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INTERNET RADIO AND BROADCAST METHOD 
TECHNICAL FIELD 

This invention relates to Internet media data streams and the like, and more particularly to a copyright-compliant 
audio/video/radio broadcast system over the Internet where each individual user is able to set his or her preferences 
regarding works played so as to influence the frequency such works are broadcast to the user. 

BACKGROUND ART 

The rise of the Internet has provided many different channels through which media can be presented to users. 
RealNetworks' RealMedia, Apple QuickTime, and Windows Media all provide players through which live or 
previously-recorded data streams can be displayed, played back, or broadcast to the individual user. Both audio and 
video are generally available through these programs and provide a higher and more attractive degree of interactivity 
with the Internet. 

Regular radio broadcasts are based upon a central individual or station broadcasting songs, or other audio 
information, electromagnetically. Different radio stations are separated by their different carrier frequencies. Amplitude 
modulation (AM) and frequency modulation (FM) provide two means by which radio broadcast can be effected by a 
transmitter to a receiver. If an individual wants to affect the songs that are played by the radio station, he or she may 
write, call, fax, e-mail, or otherwise transmit their preferences to the radio station. 

However, one person's preferred music may not be as appreciated by another individual. Music can be very 
personal, often affecting a person at an emotional level. When the radio station broadcasts a song or other audio signal, 
all receivers tuned to the carrier frequency pick up the broadcast and either enjoy or suffer the broadcast equally. 

It would be much more advantageous to allow each individual to influence, their own set of song playlists. 
Currently, this is not achievable by wireless broadcast means. However, unique data stream addressing available 
through Internet data processing might provide means by which an Internet radio could be advantageously affected. 
Other Internet broadcasting processes are known, but generally follow the known radio station format of broadcasting a 
single song, or data stream, to all users tuned to the station or channel In compliance with the Digital Millennium 
Copyright Act (DMCA), such a radio would have to comply with statutory regulations regarding the broadcast of songs 
and would generally have to avoid the role of an "on-demand" system, as this might be in violation of statutory 
regulation. 

The following patents may have some bearing on the art relevant to the present invention: 



U.S. PATENT NUMBER 


INVENTOR 
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6,052,717 


Reynolds et al. 


April 18,2000 


6,038,591 


Wolfe et al. 


March 14, 2000 


6,031,797 


Van Ryzin et al. 


February 29, 2000 


6,026,439 


Chowdhury et al. 


February 15,2000 


5,987,525 


Roberts et al. 


November 16, 1999 


5,945,988 


Williams et al. 


August 31, 1999 


5,930,768 


Hooban 


July 27, 1999 


5,864,868 


Contois 


January 26, 1999 


5,819,160 


Foladare et al. 


October 6, 1998 
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U.S. PATENT NUMBER 


rNVENTOR 


DATE OF ISSUE 


5,809,246 


Goldman 


September 15, 1998 


5,790,423 


Lau et al. 


August 4, 1998 


5,758,257 


Herz et al. 


May 26, 1998 


5,740,134 


Peterson 


April 14, 1998 


5,726,909 


Krikorian 


March 10, 1998 


5,721,827 


Logan et al. 


February 24, 1998 


5,661,787 


Pocock 


August 26, 1997 


5,616,876 


Gluts 


April 1, 1997 


5,592,511 


Schoen et al. 


January 7, 1997 


5,539,635 


Larson, Jr. 


July 23, 1996 



DISCLOSURE OF INVENTION 

The present invention provides a copyright-compl iant, broad-based, individually-tai lored Internet med ia broadcast 
system and method. The present invention provides means by which users may individually rate or indicate music, 
music videos, or other recorded media that they enjoy hearing from a vast musical or other database. Additionally, such 
users may also indicate the exclusion of music/media that is to their distaste. In so doing, the user interaction is limited 
to that decision-making role that is necessary for the user to establish his or her preferences. The Internet radio of the 
present invention and its method take care of the rest, providing the end user a media or radio channel tailored to his or 
her own musical tastes. In this way, the present invention can be said to "microcast," or "narrowcast" the content of 
personalized songlists to individual listening stations or users. As the broadcast uses Internet protocol, each data packet 
of each data stream has its own individual address, namely, the end-user's data stream player. As the present invention 
is scalable, thousands, even tens or hundreds of thousands of listeners can be handled by the present invention. With the 
advance of data-transmission technology, tens or hundreds of millions of users may be served by, or given access to, a 
system incorporating the present invention, including the delivery of user-preferred data streams by wireless 
communication links. 

Mention is made herein of the present invention with respect to music broadcast to provide a personalized 
Internet, or data stream, radio. Note should be taken that use of the term "radio," "music," and the like includes any 
recorded datastream content, including music videos and the like. 

At the core of the present invention is the playlist generator. It is the generated songl ist that is associated with the 
user's account and indicates to the system which song is to be played next. Once a song has been selected, it is then 
streamed as data out to the individual's computer (uniquely identified by Internet protocol). As the centra! server of the 
system can handle a large number of users at any one time, it becomes possible to serve each user with his or her own 
individual data stream. In this case, the data stream comprises audio and/or video information and serves to establish a 
situation similar to each user having his or her own individual radio station that he or she programs. The list can be 
created in advance and stored, or generated, in real time when needed. Collaborative filtering techniques may be used in 
constructing the playlist. 

Other applications for the present method may also exist when similar circumstances are present where a large 
database of information is available that is subject to individual preferences. In a broad sense, the present invention 
provides means by which individual subsets of an all-encompassing data space may be defined, modified, and preserved, 
subject to a variety of influences and allowing some serendipitous, or random, events to occur. 
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BRIEF DESCRIPTION OF DRAWINGS 

Figure 1 is a schematic view of the system architecture used to achieve one embodiment of the present invention. 
Figure 2 is a screen shot showing a computer desktop with the audio player and user homepage for the present 
invention. 

Figure 3 is a screen shot showing a computer desktop with the video player and user homepage for the present 
invention. 

BRIEF DESCRIPTION OF APPENDICES 

The following appendices are incorporated herein by this reference thereto. 

Appendix 1 is an excerpted text listing of a playlist generated in conformance with the present invention. 
Appendix 2 is a source code listing for one embodiment of the present invention. 

M QDE(S) FOR CARRYING OUT THE INVENTI ON 

The detailed description set forth below in connection with the appended drawings is intended as a description of 
presently-preferred embodiments of the invention and is not intended to represent the only forms in which the present 
invention may be constructed and/or utilized. The description sets forth the functions and the sequence of steps for 
constructing and operating the invention in connection with the illustrated embodiments. However, it is to be understood 
that the same or equivalent functions and sequences may be accomplished by different embodiments that are also 
intended to be encompassed within the spirit and scope of the invention. 

This patent application is related to United States Provisional Patent Application Serial Number 60/164,846 filed 
November 1 0, 1999 for Internet Radio and Broadcast Method, which application is incorporated herein by this reference 
thereto. 

As mentioned above, use of the term "radio," "music," and the like includes any recorded datastream content, 
including music, videos, recorded sports events and concerts, and the like. 

In Figure 1, the general structure of the present system is shown where the LAUNCHcast Player provides user 
feedback and indication of song preference through Java Servlets and JavaScript code. In one embodiment, a Windows 
Media Player may provide the interface allowing the audio and/or video broadcast to take place at the user's computer. 
Other media players now known or developed in the future may also suffice and operate to good advantage. Mentioned 
use of the Windows Media Player system is to be considered as indicating any appropriately functioning media player. 
Song or video information is available through both the player and the accompanying data window. 

Referring now to Figure I , the architecture and system structure of the Internet radio and broadcast method of the 
present invention is shown in schematic form. The system 100 is generally focused upon the player 102. The player 102 
is the component that the user sees and is ultimately the arbiter of the media datastream service provided by the present 
invention. As shown in Figure 1 , the player 102 has a song information section 104, a rating tool 106, and a player 108. 
For this last component, the player 108 is indicated as being a Windows Media player. However, other media players 
can also be used to good advantage in order to achieve the present invention. 

Through its components, the player 102 is linked or associated to a number of other sources of information and 
programs, including Java or other servlets. The present invention, when implemented in software, may be so 
implemented using Java-family of computer program languages. A servlet is Java programming that runs as a part of a 
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network service, such as an HTTP server, in response to requests from clients. In this case, the client can be considered 
to be the player 102 while the HTTP server can be the servers for the database 160 and the media content library 180. 

At a center of the present invention is the player 108. The player 108 allows the content to be broadcast to the 
individual user and serves as means by which the user can enjoy such content In addition to being linked to the media 
database 180, the player 108 is also in communication with a media gateway servlet 120 and a play list generator servlet 
122. As discussed in more detail below, these two servlets provide the player the ability to play streaming media in 
conformance with the present invention. 

The rating tool 106 is coupled to the database 160 via a rating acceptor servlet 130 and a ratings cache servlet 
132. As indicated in Figure 1, the rating acceptor servlet 130 and ratings cache servlet 132 are also in communication 
with one another, as set forth in more detail below. 

The song information component 1 04 of the player 102 may provide links to other information available through 
the database 1 60 or otherwise. For example, the song information tool 1 04 may provide links to other user pages 1 40, a 
station manager 142, provided home pages of various artists 144, as well as links to album pages 146 of such artists or 
otherwise. Additionally, a central homepage 148 may be present that allows travel or linking to any or all of available 
pages or services. 

Note should be taken that the database 160 is not necessarily the home for the media-library 180. In fact, 
according to present technology, it may be more advantageous to provide some means by which high-speed access can 
be provided to the media library 180. By separating the database 1 60 from the media library 1 80 faster and better service 
may be provided to users so they may enjoy the content of datastream better. Certain infrastructures may allow for 
offsite residence of the media contained in the media library 180. Pointers or other indicators to such information in an 
indexed or other form can thereby provide the link necessary to deliver the preferred or indicated content by the user 
from the media library 180 to that same user. 

As shown in Figure 1, the database 160 may hold a variety of types of information, including: user data 162, 
play lists 164, and song data 166. Such information is stored by the database 160 and updated by the servlets as set forth 
in the present invention, including the user code set forth in Appendix 2. 

In Figure 2, the player, or playback, window 102 is shown and is highly interactive with several embedded 
hyperlinks. In the upper right-hand corner of the playback window 102, the indication of "asjordan" is made. By 
clicking on this link, more information about the current station may be given and/or the ability to change such station. 
The user's page 140 may be activated and shown upon clicking the usemame link. In the right center of the playback 
window, a "RATE -IT" window indicator that is the rating tool 106 is given, allowing the individual to rate the current 
"SONG", the "ARTIST 1 performing the current song, and/or an "ALBUM" containing the song. Below the "RATE IT' 
indicator, hyperlinks to "RECENT SONGS" "BUY", and "STATION MANAGER" are present allowing the user to 
travel to those destinations and either learn more information, purchase or review purchasing information about the 
current album being played, as well as access the station manager for the present invention. 

Below the song information window 104, icons are given for Play/Pause, Skip This Song, Skip This Song and 
Never Play It Again ("Delete"), and a Volume control. The question mark ("?") shown below the "Song Information 
area" window is a hyperlink to a Help file for the playback window 102 and the Internet Radio system of the present 
invention. These icons are also shown in the other playback window Figures, such as that for the video playback user 
interface/client 102 shown in Figure 3. 

Figures 2 and 3 show a desktop display of the system 1 00 in action from the user's point of view. A tool tip may 
be given when the cursor hovers over the song title. The same may be similarly true for the artist and the album 
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currently playing. Note should be taken that just as the song rating indicator is highlighted and active in the middle right 
section of the playback window, the song title is highlighted in the upper portion of the playback window. 

Additionally, the left and center middle portion of the playback window provides information regarding fans who 
have strong positive feelings about the present song, artist, and/or album, as well as an average rating for all users or 
some subset of users on the system. 

Figures 2 and 3 show small balloons on the right-hand side of the central dark area across from the "Fans." These 
balloons may have a letter "W" inside of them to indicate another listener is currently online and can be engaged via the 
instant messaging ("whisper") function. Figures 2 and 3 also show graphic information that may be used for advertising 
or other hyperlinks. In generating the playlist of the present invention, the user can be informed as to why a particular 
song was picked. 

For other links and presentation of information in the player 102, a tool tip may be presented when the cursor 
hovers over an area. A tool tip is a small window providing succinct information about the item under the cursor when 
the cursor hovers over that item. 

When the system 100 is updating and obtaining a new data stream from the system for the user, a display may be 
given to the user to indicate ongoing activity of the playback system. Such visual activity in the form of animation 
assures the listener/viewer that the short span of silence, or "dead air," following a song is only temporary and that a new 
song will soon play. Generally, in order to promote interactivity and to take advantage of the new media that the Internet 
provides, the windows shown in the Figures 2 and 3 contain ample internal hyperlinks that lead to web pages providing 
information regarding music, artists 144, and/or their works 146, web pages regarding other users of the system (as DJs 
or otherwise) 140, and/or web pages regarding the user's control of the system (preferences, etc.) 142. 

The default paradigm for the user interface/player 102 is to allow the user the greatest degree of freedom in 
expressing preferences and in obtaining that preference information regarding music artists, and their 
publications/albums. In this way, the user's experience is enhanced as he or she hears more of the music he or she likes. 
Access to purchasing web sites is also made available where users may purchase artists' works. 

In implementing the present invention in software, the accompanying source code (Appendix 2) may be used to 
achieve the present invention. Such code is subject to copyright protection and is owned by LAUNCH Media, Inc. of 
Santa Monica, California. 

The generation of a proper playlist combining available user ratings and a media database forms an important part 
of the present invention. One such playlist as generated by the present invention is shown in Appendix 1 and is an 
excerpted form for purposes of explanation. Entries in the playlist have been removed so that the playlist may better 
serve the explanatory purposes herein without undue length or the sacrifice of sufficient detail. 

Playlist generation occurs when a user launches his client player 102. A Windows Media or other player 108 is 
embedded in the user's client player 102. The player 108 opens a call to the playlist generator servlet 122 as executed by 
the PlaylistGeneratorServlet routine (Appendix 2, page 158). The expected output from this HTTP call is an ASX 
playlist file, which in the present invention is I ist of pointers to a script that reads the actual playlist data object from the 
database 160. 

The playlist generator servlet 122 parses the particular parameters for this ASX playlist as follows: 
Object: GeneratorParameters; 

userlD: (required) the user for whom the playlist is generated; 

djID: (default is userlD) the user whose profile will be used to generate the playlist; 

moodID: (default is none) a mood which is a subset of a profile may be indicated and used to alter the preferences 
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in the playlist and under which to listen (optional); and 

bandwidth: (default is 28.8k, if not read from the user's preferences in the database) the bit rate at which the user 
wishes to listen. 

The database 160 with the playlist database 164 is checked for an existing playlist by PlaylistStatus (Appendix 2, 
page 192). If a playlist already exists, it can be used it if all the following are met (and PlaylistStatus.isStaleO returns 
false): 

ail of the parameters (userlD, djID, etc) match; 
there are more than 8 songs left; 

the newRatingsCount (counter of new personalization data since last refresh) is less than 15; and 
the playlist is less than a week old. 

If all these conditions are met, the dates for the last time the user listened to an ad, news bit, and tip may be reset 
and the playlist may be resaved. The ASX file is written out and media player begins to execute by making requests to 
the media gateway 120 to play music. ■ r ■ 

If the old playlist cannot be used, a new one is created with the playlist generator via PlaylistGenerator.createO. 

The first step is to retrieve the user's preferences via PIaylistGenerator.getOptions(). In response the following 
options are returned: 

unratedQuota: how much new (not rated) music they want hear in their playlist. The options here are 90, 80, 70, 
50, 40, 30, and 20 percent. The default is 50 percent. 

explicit lyrics: Does this user want us to play music with explicit lyrics? True or false. 

bandwidth: if the bandwidth is not already specified in the generator parameters, it is read from stored data. 
Currently, bandwidth options include 28.8, 56, and T 1 /LAN. The default is 28.8 if a valid setting of "none" is found in 
the database. 

A list of all the possible songs available for play (via PlaylistGenerator.gatherMediaO) as well as some other data 
about those songs is obtained. This is generally done using multiple threads running at the same time for better 
performance. The list of songs is held inhashtable (as via the Population subroutine (Appendix 2, page 198)). 

The database 160 is first called to load a history of all the songs played for the user in the last 30 days. This is 
stored in the database as a long string, formatted as: H <Date>=<songID>,<Date>=<songID>, ..." For performance 
reasons, reading one string from the database is faster than reading potentially several thousand rows individually from 
the database. Dates older than 30 days are ignored and the last time a song was played overwrites previous plays of a 
song. Each time a song is played via the media gateway 120, this string is appended. 

After the history loading is complete, a random integer is picked from 1 to 10. If the value is 1 , the date and 
songID string is recreated and rewritten to the database. This cleans up the string by removal of songs that were played 
more than 30 days ago as well as duplicate entries for the same songID. 

The history loads as a thread, and another database call is made to get the user's, or DJ's, list of subscribed DJs, 
genres, and radio stations (via PIaylistGenerator.getSubscriptions()) for the specific mood requested. The result of this 
call is three lists called DJs, genres, and stations. 

Once the subscriptions are available, the ratings are obtained via GetRatings. This is also done in a thread. The 
song hashtable, another hashtable that contains Artist and Album ratings (ItemsProfile); the DJ, and the list of subscribed 
DJs are all passed to the GetRatings method routine. 

A retrieval list of users whose ratings are to be retrieved is compiled using the subscribed DJs and the DJ 
requesting the playlist. A request is made to the ratings cache to retrieve all these ratings via RatingsCache.getRatingsQ. 
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When the play list generator has al I the ratings, it is ready to assemble them into categorized data structures, based 
on the properties of each rating. It iterates through al! the ratings and stores them in the following manner. If the ID of 
the user is the DJ and the rating is 0 (an 'X' in the end-user interface), the song is added to song hashtabie (via 
Population) as an "Excluded" type, meaning that song should never be played. The rating is also added to the average 
rating for songs by that artist. If the rating is not 0, the song information cache is immediately checked via 
SonglnfoCache.get() for data about this song. If the data does not exist in the cache, it is a song that was rated, but is not 
available for play (as possibly not encoded), and the song is immediately marked as an "Excluded" song. 

If all of the above tests pass, the song is added to the song hashtabie with a type of "Explicit". The rating for the 
song is included in the calculation of this DJ's average rating of songs by the artist 

Each song that is rated by subscribed DJs is added to the song hashtabie. The subscribed DJ's rating for the song 
is included in the calculation of the subscribed DJs 1 average rating for this song. 

For albums, the ratings profile is obtained from the item rating profiles. If a ratings profile for an album does not 
yet exist, then the data regarding the album is retrieved and a ratings profile is created. 

If the rater is the user requesting the playlist, the rating for this item is set to the user's rating. However, if the 
rater is a subscribed DJ, the rating is added to the DJ's average for this album. 

For artists, the rating procedure is the same as for albums, except any ratings made for the artists listed as 
"Various Artists", "Soundtrack", or "Original Soundtrack" are discarded or ignored in the relevant calculations. 

The top 1000 most popular songs (via PlaylistGenerator.getPopuIarO) in the bandwidth type specified may be 
added to the song candidate hashtabie. This popular list is maintained in the song information cache. Before each song 
is added to the song hashtabie, inspection is made to see if the song is already in the candidate hashtabie (perhaps put 
there by another query), if so, inspection is made to make sure that the song is not of type "Excluded", or the song is 
discarded. If the song is added to the song hashtabie, it is added under the type "Unrated". 

A maximum of 5000 songs are picked randomly (via PlaylistGenerator.getRandom()). Initially, a count is made 
of the number of songs contained in each and all of the genres a user has selected (via SonglnfoCache.countlnGenresO). 
Songs may be in multiple genres. The number of songs is then divided by the total number of songs in the song 
information cache. If the result is less than 5%, songs are picked directly from a list of songs only in those genres. 
Otherwise, songs can be picked randomly from all available songs. This calculation may be performed to avoid the 
situation where a user has selected a small number of genres and picking songs randomly will return only a few songs 
that are available or allowable for play when considering their genres. 

In order to select songs only from selected genres, a determination is made of the total number of songs to pick 
(via totalToPick) from the lesser of 5000 and the total number of songs in the selected genres. For each genre, a copy of 
the list of songs in that genre is obtained from the song information cache (via SonglnfoCache.getlnGenreO). The 
number of songs to pick from each genre is determined from the following formula: songs to pick = totalToPick * 
(number of songs in this genre / total number of songs in the selected genres). 

The determined number of songs is picked and attempts are made to add the songs to the song hashtabie with a 
type of "Unrated". A song is not added if it is already in the hashtabie. 

In order to select from all songs, a song is randomly selected 5000 times. Each time, attempts are made to add the 
song if it is not already there as picked, as described above. Once the process finishes adding random songs, all the 
ratings for the songs are retrieved as are all the dates of when the songs were played for the user. The explicit, implicit, 
and unrated lists built in the last step are taken and ordered in descending order by score, or rating, using a quicksort or 
other algorithm. 
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The number of songs to pick from each list is determined. For example, if the size of a play list is 50 songs, the 
following may occur. If the user is listening to his own station, the following formula may be used: if the user's list of 
explicit and implicit songs is smaller than 100 songs, 90% of the songs must be picked from the unrated list to avoid 
playing the user's rated songs too much. The user's unrated quota may, then, be set to 90. Otherwise, an unrated quota 
may be used from the user's stored options. 

Under some circumstances the maximum number of songs available from the explicit and implicit song lists is 
calculated as follows: 

maximumRated = playlistSize * (100 - unratedQuota) * 0.01. 

The maximum number of songs available from the explicit list may be calculated as: 

MaximumExplicit = number of songs in the explicit list * .20. 

A number of songs to pick from the explicitly-rated list may then be: 

explicitToPick — playlistSize * (100 - unrated quota) * 0.01 * (number of songs in the explicit list / sum of 
explicit and implicit songs) * 3); 

From this the number of implicit songs is simply: 

impiicitToPick = maxiumum Rated - explicitToPick. _ -.J 

Confirmation can be made to ensure that more explicit songs have not been picked than indicated by 
maximumExplicit and that no more implicit songs have been picked than those that are in the implicit list. The number 
of unrated songs is then: playlistSize - (explicitToPick - impiicitToPick) 

If the user is listening to a station other than his own and the number of songs in the explicit and implicit list total 
greater than 200, then the following calculations are made: 

explicitToPick = Minimum(playlistSize * .50, 20% of explicit songs); and 

impiicitToPick = Minimum(playlistSize, # of implicit songs) - explicitToPick 

If, for some reason, a sufficient and/or playlistSize number of songs is not obtained from this calculation, a third 
of the songs is picked from each of explicit, implicit and unrated songs with a check to ensure that not more than 20% of 
the songs on the rated and unrated lists are picked. As a fallback measure if none of the methods above used to calculate 
the number of songs to pick worked, the songs are selected as a third of the playlistSize from each list, making sure not 
to pick more than 20% of the rated and unrated lists. 

A list of albums and artists from and by which songs have been played for this user in the last 3 hours is copied or 
otherwise made available to the process set forth herein and the songs for this play list are picked via 
PlaylistGenerator.pickSongsO- A list of all the picks needed is made (via PickList). For example, if there is a playlist of 
50 songs, the list may contain 10 entries for explicit songs, 20 for implicit songs, and 20 for unrated songs. 

While there are still songs to pick, iteration is made through the following cycle: 

a. randomly pick a song list type (explicit, implicit, unrated) with a probability based on the proportion of songs 
to come from each list; 

b. pick a random song index from that list (which has already been sorted in descending order of score), based on 
the following formula (via SongGroup.pickRandom()): 

sizeOfList = the number of songs in this list; 

random = a randomly-chosen number between 0 and (sizeOfList - 1) + 0.01 ; and 
index of song to pick = ((rand A 7) / sizeOfList - 1 A 7) * (sizeOfList - 1)). 

This formula allows songs to be picked somewhat randomly, while guaranteeing a high probability that the song 
picked will come from highest scored. The higher the ranking of the song in the score matrix, the higher the probability 
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it will be picked. This algorithm scales well for any size of list because it is rank-based, not just score based. 

The song at that index is removed from the list. If for some reason a valid song is not obtained (possibly the song 
list already exhausted), another song is added to the list of types to pick of this type. 

Once a song is picked, its album and artist information are obtained. 

If the artist is not a "Various Artist" and the sum of the number of songs played by this artist and already picked 
for this playlist by this artist is greater than or equal to 3, this song cannot be played under the RIAA (Recording Industry 
Associates of America) and/or DMCA (Digital Millennium Copyright Act) rules. Other rules may also be implemented 
in the present invention to accommodate statutory and other rights and/or restrictions. 

The song is marked as "rejected" and another song is added to the list of songs to pick from the same list the 
rejected song was picked from. The same test is performed for albums, with the maximum played, for example, being 2. 
If the song was picked successfully and was within legal or other boundaries, the number of songs picked from this 
album and by this artist is incremented. The song is added to the final list of songs for the playlist and the order in which 
the song was picked for the playlist is marked, or noted. 

If, for some reason, a playlistSize number of songs is not obtained, the existing playlist is deleted and popular 
songs are added to the song hashtable, and the song lists are re-sorted and re-picked ignoring the user's genres selections. 

The picking of news clips is done simply by picking a specific number of unique news items that are in the 
specified bandwidth format A list of available news clips is stored in the song information cache. Ads may be picked in 
the same way as news clips are picked. However, a difference may be present in the different numberof ads to pick. 
Tips may also be picked in the same manner as news clips, with a different number of tips to pick. 

The order of the songs may be randomly shuffled in the playlist and the playlist may be serialized and saved to 
the database. Finally, the ASX file may be returned to the player 108. 

Every 5 minutes, the player ] 02/1 08 "pings" the Playlist Generator 122. If the playlist is stale or has 8 songs or 
less left in it, the playlist generator regenerates the playlist and replaces the one previously saved in the database. .• 

As an additional enhancement to the present invention, playlists from commercial and other radio stations 
throughout the United States, and elsewhere, are made available so that playlists may be affected by such radio stations 
and by popularity of particular musical works. 

In achieving the Internet radio of the present invention, a rating acceptor 130 in the form of the 
Rating WidgetServlet routine (Appendix 2, page 222) takes HTTP requests to rate and gets ratings for songs, albums, and 
artists. When a rating is saved, it written to the ratings database and if the user who rated the item is designated as being 
in the ratings cache, the rating change is added to the queue of ratings updates. 

Once every minute, the ratings updates are sent to all the ratings caches that have registered their IP address in the 
database. Every hour, the list of ratings caches are retrieved from the database. Every ten minutes, the list of users in the 
cache are retrieved from the database. 

The song information cache is implemented through the SonglnfoCache routine (Appendix 2, page 265) and may 
be a large in-memory cache of relatively static data that is used in playlist generation. It may include a list and hashtable 
of all songs which includes identifying numbers, media formats available, average rating, artist and album information, 
explicit lyrics mark, genres the song is in, and radio stations that play the song. Also, other information may be included 
in the song information cache, including: a hashtable of artist information;, a hashtable of album information; a list and 
hashtable of all ads including identifying numbers and media formats available; a list and hashtable of all news clips 
including identifying numbers and media formats available; a list and hashtable of all audio tips including identifying 
numbers and media formats available; a lists of the 1000 most popular songs in each media format; lists of ail songs in 
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each genre; and a cache of frequently-accessed ratings profiles. This last cache is seen in the RatingsCache 1 32 routine 
(Appendix 2, page 211). The song information cache is completely rebuilt once a day from the database. 

The ratings cache caches the entire ratings profile for the top 100 users who are known to be accessed frequently. 
The ratings cache is implemented through the RatingsCache routine (Appendix 2, page 21 1). On startup, the ratings 
cache registers its IP address in the database to subscribe to ratings updates. These users are typically DJs (users with 
broadcasted or subscribed ratings) that have many subscribers, or users who simply use LAlTNCHcast frequently. Each 
ratings cache recalculates the most frequently-accessed users and writes it to the database every 8 hours. At that time, 
the entire cache is discarded and reread from the database to erase any lingering corruption. Each ratings cache checks 
the database every 10 minutes for changes in the list of users to be cached and updates the ratings cache as appropriate. 

Note should be taken that many of the parameters set forth herein are discretionary and advisory. Consequently, 
those properly and legitimately implementing the present invention may alter such parameters, such as when events 
occur and event timing as above, according to system operation preferences. 

For each user who is not in the ratings cache, their ID is appended to a list of users whose profiles need to be 
retrieved from the database 160. Users who have been added to the cache recently have their profiles added to the list of 
ratings to be returned to the PlaylistGenerator 122 routine (Appendix 2, page 1 58). All non-cached users' ratings are 
retrieved from the database 160, are appended to the list of ratings, and are returned to the PlaylistGenerator 122. The 
album and artist ratings are retrieved in a separate query from the song ratings. Each runs in its own thread in parallel for 
optimal performance. 

The media gateway 120 is a Java servlet that brokers the relationship between the end user's (Windows Media) 
Player 108, the database 106, and media library, or Windows Media Server, 180 and logs all media access. The 
MediaGatewayServlet routine (Appendix 2, page 1 12) performs this function. Because the client's Windows Media 
Player play list (.sax file) does not contain any information about the actual songs or ads in the user's play list, the media 
gateway 120 contains the logic described below to redirect the user's player to the correct media address on the media 
library 180. 

For security reasons, the media gateway 120 may check to see that the client 102 is accessing it from the 
Windows Media Player client 108 (and not a web browser or other application). If not, it may redirect the user to an 
error media file. The media gateway 120 then pulls the user's ID off the query string and retrieves that user's playlist 
object from the database 160. The gateway 120 inspects timestamps in the user's playlist object that indicate when the 
user last heard an ad, tip, song or other media item and determines if it is time to insert an ad, tip, or news item in the 
datastream, or just play the next song. 

If the user has not heard an ad, for example, for a pre-defined period of time, the media gateway 120 resets an ad 
timestamp and retrieves an ad path from the user's ad playlist and passes that MMS (Microsoft Media Server) redirect 
instruction/address to the end user's Windows Media client 108. If no ad is available, the process continues and plays the 
next song in the user's playlist. If it is not time to play an ad, the timestamp is checked to see if it is time to play a tip. 
The process then follows the same logic, above, for ads to retrieve and play a tip, instead of an ad. If it is not time to 
play an ad or tip, the timestamp is checked to see if it is time to play a news item. The process then follows the same 
logic as for ads to retrieve and play a news item. 

If it is not time to play an ad, tip, news item, or other stream (the usual case), the media gateway 120 retrieves the 
path of the next song in the playlist and returns that address via an MMS redirect to the client's Windows Media Player 
108. In all cases, the medialD of the ad, tip, or song played is logged in the database 160 under that user's ID. This 
logging information is used to display what the user is listening to on the user's station page and under the "Who's 
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Listening" page. These pages may be associated with the central home page 148 in a manner similar to that of the user 
pages 140 as history data in the play list generator, and in calculating a Top 100 chart for the most popular songs and/or 
streams. 

While there may be some preference for an "on-demand" service such that individuals may pick their own radio 
playlists, the element of randomness and pleasant surprise is inherent in the present invention. Additionally, statutory 
requirements prevent users from turning the Internet into their own home stereo system. "On-demand" service is 
generally prevented by statute and may be a violation of copyright. Consequently, any statutory regulations, such as the 
Digital Millennium Copyright Act (DMCA), and other limitations can be programmed automatically into the present 
invention. In so doing, the present invention complies with all applicable law and delivers to the user a musical 
experience generally aligned with his or her preferences. 

Many users often listen to music while doing programming or the like. Such music can now be delivered over the 
Internet via the user's very own radio station through the present invention. Additionally, users may select other 
individuals or DJs, to influence their musical play list just as the user does. The DJ, online or otherwise, becomes an 
additional factor in influencing the user's preferences and play list. Some individuals may act as real DJs, serving to 
provide content to an audience of subscribers through the Internet Programs of special interest may also be developed 
and subscribed to by listeners using the present invention. Through the heavily hyperlinked (but easily understandable) 
interface set forth in the Figures and described above, a user may establish musical (or other data stream) preferences. In 
establishing such preferences, the music played to the listener is tailored to that listener and provides an enhanced 
musical experience on an individual basis. 

While the present invention has been described with reference to a preferred embodiment or to particular 
embodiments, it will be understood that various changes and additional variations may be made and equivalents may be 
substituted for elements thereof without departing from the scope of the invention or the inventive concept thereof. In 
addition, many modifications may be made to adapt a particular situation or material to the teachings of the invention 
without departing from the essential scope thereof. Therefore, it is intended that the invention not be limited to particular 
embodiments disclosed herein for carrying it out, but that the invention includes all embodiments falling within the 
scope of the appended claims. 

INDUSTRIAL APPLICABILITY 

It is an object of the present invention to provide individualized data stream programming according to an 
individual's preference. 

It is yet another object of the present invention to provide an Internet-based radio or music playing system that is 
biased according to each user's preferences. 

It is yet another object of the present invention to provide a means by which song playlists may be generated for 
such an Internet radio. 

It is an object of the present invention to provide copyright-compliant media streams for Internet and other 
networked systems broadcast. 

These and other objects, advantages, and the industrial utility of the present invention will be apparent from a 
review of the accompanying specification and drawings. 
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Playlist status for userlD 6474126: 
newRatingsCount: 0 
moodID: 0 
djlD: 6474126 
songsRemaining: 50 
mediaType: 212 

generating because forceRefresh is on 

regenerating playlist with parameters: userID=6474126, bandwidth=28.8k, moodlD=0, djID=6474126<PRE> 
start of createP lay list 

0.0 lap time, 0.0 total . 



starting gathering threads at 
0.0 lap time, 0.0 total 



GetLastPiayed loaded 618 dates 
getS ubscript ions done 

0.063 lap time, O.063 : total- — — 



All threads started 



0.0 lap time, 0.063 total 



getPopular done 

0.047 lap time, 0.11 total 



getRandom done (picked 5000 songs) 
1.281 lap time, 1.391 total 



genres for mood 0:64,44,46,48, 50,45, 47,49,51,63,67,1,0,6, 7, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19,21,22, 

23,24,68,69,73,74,75,76,77,78,79,80, 
gatherMedia done 



0.0 lap time, 1.391 total 



scores calculated 



0.156 lap time, 1.547 total 



recently played albums and artists marked 
0.0 lap time, 1.547 total 

Of 6749 songs, these are the reasons for exclusion: 546 were already excluded, 349 were not encoded, 34 were 

played in the last 3 hours, 0 had explicit lyrics, 0 were not in mediaType 212, 1292 were not in their genres, 
482 had an implicit rating of 0. 

There are 4046 songs available for play 

ordering... 



WO 01/35667 
0.0 lap time, 1.547 total 
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finished sorting vectors at 



0.11 lap time, 1.657 total 



Available: explicit songs: 388.0, implicit songs: 2334.0, unrated songs: 1 324.0 
Ratio: 20 

Picking: explicit songs: 17, implicit songs: 23, unrated songs: 10, method = Unrated Ratio 
start of pickSongs 



0.0 lap time, 1.657 total 



end of pickSongs 



0.0 lap time, 1.657 total 



picked news 



0.0 lap time, 1.657 total 



picked ads 



0.015 lap time, 1.672 total 



picked tips 



0.0 lap time, 1.672 total 



playlist has 50 songs 
shuffling playlist... 
end of createPlaylist 

0.0 lap time, 1.672 total 



starting to save playlist 



0.016 lap time, 1.688 total 



done saving playlist 



0.031 lap time, 1.719 total 



</PRE> 
<PRE> 

Playlist 0 for userlD 6474126 (djID 6474126) in mood 0 with mediaType 212, pickCounts: explicit to pick: 17, 

implicit to pick: 23, unrated to pick: 10 has 50 songs: 
37409 146690 1022473 1364151 Emitt Rhodes Listen, Listen: The Best Of Emitt Rhodes You're A Very Lovely 

Woman - The Merry-Go- Round) 
37718 43307 1016600 385563 Madonna Erotica Erotica 
45680 43305 1016600 385517 Madonna The Immaculate Collection Cherish 
40237 98477 1025497 900407 Squeeze The Piccadilly Collection * Loving You Tonight 
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21825 132410 1 027798 1 2 1 2736 U2 The Best Of 1 980- 1 990 [Limited] New Year's Day 
37268 137097 1028125 1259519 Various Artists Made On Earth Untitled - Total Eclipse 
8405 41860 1015576 3725 19 The Lightning Seeds Sense Sense 

3 1 547 9 1 874 1 0 1 5450 839523 Jackie Leven Forbidden Songs Of The Dying West Birds Leave Shadows 
42209 100072 1028125 1407544 Various Artists Assemblage Vol 1 Taksu - Lights in a Fat City 
39401 105661 1005547 956525 Paula Cole This Fire * Tiger 

52454 85650 1024526 778897 Carly Simon Clouds In My Coffee 1965-1995 [Box] Stuff That Dreams Are Made 
Of, The 

53486 5 II 28 1021 142 458446 Pink Floyd Ummagumma Narrow Way Part 1, The - David Gilmour 

17982 58282 1025027 526886 Social Distortion Prison Bound Backstreet Girl 

22578 14393 1000398 123761 Bryan Adams So Far So Good Summer Of '69 

6947 130669 1009757 1 193855 Fun Lovin' Criminals 100% Columbian * Big Night Out 

39632 1 13337 1028125 101 1924 Various Artists Pure Moods Crockett's Theme - Jan Hammer 

30674 93944 1028256 857682 The Verve Pipe Villains * Cattle 

28189 61860 1026856 559756 They Might Be Giants They Might Be Giants Toddler Hi way 

16788 23890 1005543 212417 Jude Cole Start The Car Right There Now 

37247 137097 1028125 1259512 Various Artists Made On Earth Portnawack - Typhoon 

28606 64190 1030389 578647 Vanilla Fudge Rock & Roll Windmills Of Your Mind, The - (original mix) 

6299 1 18154 1005865 1062093 Cornershbp When I Was Born For The 7th Time * Brimful Of Asha 

29369 74082 1025801 673069 Sting Fields Of Gold: The Best Of Sting 1984-1994 They Dance Alone (Cueca Solo) 

23334 148558 1026856 1386237 They Might Be Giants Miscellaneous T Kiss Me, Son Of God - (alternate version) 

53363 50728 1021 142 454344 Pink Floyd A Saucerful Of Secrets Let There Be More Light - - - --■ — - 

50557 50901 1020983 455893 Tom Petty Into The Great Wide Open All Or Nothin' 

42791 142342 1 025039 1 3274 1 6 Soft Cell Non-Stop Ecstatic Dancing Insecure Me 

30719 95006 1021869 867248 R.E.M. New Adventures In Hi-Fi Wake-Up Bomb, The - (live) 

42923 148836 1015285 1388605 Ben Lee Breathing Tornados * Cigarettes Will Kill You 

39860 123837 1018539 1 122003 Morcheeba Big Calm Friction 

30644 93944 1028256 857672 The Verve Pipe Villains * Drive You Mild 

31529 91874 1015450 839517 Jackie Leven Forbidden Songs Of The Dying West Working Alorie/A Blessing 

39320 92012 1028514 841099 Loudon Wainwright III Grown Man Human Cannonball 

22344 143220 1000012 1331978 10,000 Maniacs The Earth Pressed Flat * [4/20] Hidden In My Heart 

26698 47344 101 8869 423656 Peter Murphy Should The World Fail To Fall Apart God Sends 

21660 130952 1021402 1 1 96259 Portlshead PNYC * Strangers 

26686 47344 101 8869 423652 Peter Murphy Should The World Fail To Fall Apart Light Pours Out Of Me, The 

39137 87489 1023065 798733 David Lee Roth The Best Li!' Ain't Enough, A 

7646 145523 1030217 1352144 Buddy Holly 20th Century Masters:... [4/20] Maybe Baby 

44144 25421 1006149 227025 Crosby, Stills & Nash CSN [Box] Southern Cross 

21999 135883 1038686 1242702 The Hope Blister Smile's OK... Is Jesus Your Pal 

39644 1 13337 1028125 101 1928 Various Artists Pure Moods Theme From "Twin Peaks - Fire Walk With Me" - 
Angelo Badalamenti 

505 1 5 50895 1 020983 455822 Tom Petty Full Moon Fever Face In The Crowd, A 
40510 1 17098 1018623 1049778 Morrissey Maladjusted He Cried 
31805 87741 1013181 801582 Jars Of Clay Jars Of Clay Like A Child 

29384 74082 1025801 673074 Sting Fields Of Gold: The Best Of Sting 1984-1994 We'll Be Together - (previously 

unre leased version) 
25621 36886 1012859 328927 INXS X Disappear 

28039 60022 1025830 544499 The Stone Roses Second Coming Love Spreads 

26269 41495 1015374 369132 Lemonheads Come On Feel The Lemonheads Into Your Arms 

52466 85650 1024526 778868 Carly Simon Clouds In My Coffee 1965-1995 [Box] Better Not Tell Her 

2 songs are by the artist Jackie Leven (IQ15450) 
1 songs are by the artist Bryan Adams ( 1 000398) 
1 songs are by the artist Paula Cole (1005547) 
1 songs are by the artist Soft Cell (1025039) 

1 songs are by the artist Portishead (1021402) 

2 songs are by the artist They Might Be Giants (1 026856) 
1 songs are by the artist Crosby, Stills & Nash (1006149) 
1 songs are by the artist Vanilla Fudge (1030389) 

1 songs are by the artist Jude Cole ( 1 005543) 

2 songs are by the artist Carly Simon ( 1 024526) 
2 songs are by the artist Peter Murphy (101 8869) 

1 songs are by the artist Social Distortion (1025027) 
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2 songs are by the artist The Verve Pipe (1028256) 

2 songs are by the artist Tom Petty ( 1 020983) 

I songs are by the artist The Stone Roses (1025830) 

1 songs are by the artist Fun Lovin' Criminals (1009757) 

1 songs are by the artist Morcheeba (1018539) 

1 songs are by the artist R.E.M. (1021869) 

I songs are by the artist Jars Of Clay (1013181) 

1 songs are by the artist Emitt Rhodes (1022473) 
5 songs are by the artist Various Artists (1 028 125) 

2 songs are by the artist Sting (1025801) 

1 songs are by the artist Squeeze (1025497) 
1 songs are by the artist Morrissey (1018623) 

1 songs are by the artist David Lee Roth (1023065) 

2 songs are by the artist Madonna (1016600) 

1 songs are by the artist Ben Lee (1015285) - 

2 songs are by the artist Pink Floyd (1021 142) 
1 songs are by the artist INXS (1012859) 

1 songs are by the artist Loudon Wainwright Ml (1028514) 

1 songs are by the artist U2 (1027798) 

1 songs are by the artist Lemonheads (1015374) 

I songs are by the artist The Lightning Seeds (1015576) 

1 songs are by the artist Buddy Holly ( 1 0302 1 7) - - « - ~ . . . — . 

1 songs are by the artist 10,000 Maniacs (1000012). 

1 songs are by the artist Cornershop (1005865) 

1 songs are by the artist The Hope Blister (1038686) 

1 songs are from the album The Best Of 1980-1990 [Limited] (132410) 

1 songs are from the album Into The Great Wide Open (50901 ) 

I songs are from the album Full Moon Fever (50895) 

1 songs are from the album Miscellaneous T (148558) 

1 songs are from the album Come On Feel The Lemonheads (41495) 

1 songs are from the album When I Was Born For The 7th Time * (1 18154) 

1 songs are from the album 20th Century Masters:... [4/20] (145523) 

1 songs are from the album Assemblage Vol. 1 (100072) 

1 songs are from the album Erotica (43307) 

1 songs are from the album The Immaculate Collection (43305) 

2 songs are from the album Should The World Fail To Fall Apart (47344) 
I songs are from the album 100% Columbian * (130669) 

I songs are from the album Jars Of Clay (87741) 
1 songs are from the album CSN [Box] (25421) 

1 songs are from the album New Adventures In Hi-Fi (95006) 

2 songs are from the album Forbidden Songs Of The Dying West (91 874) 
1 songs are from the album Breathing Tornados * (148836) 

1 songs are from the album PN YC * (130952) 
I songs are from the album Rock & Roll (64190) 
1 songs are from the album Start The Car (23890) 

1 songs are from the album So Far So Good (14393) 

2 songs are from the album Fields Of Gold: The Best Of Sting 1984- 1994 (74082) 
1 songs are from the album They Might Be Giants (61 860) 

1 songs are from the album Sense (41860) 

2 songs are from the album Made On Earth (137097) 
I songs are from the album Maladjusted (1 17098) 

I songs are from the album Smile's OK... (1 35883) 

1 songs are from the album Listen, Listen: The Best Of Emitt Rhodes (146690) 

1 songs are from the album Non-Stop Ecstatic Dancing (142342) 

1 songs are from the album Second Coming (60022) 

I songs are from the album A Saucerful Of Secrets (50728) 

1 songs are from the album The Best (87489) 

1 songs are from the album Ummagumma (5 1 128) 

1 songs are from the alburn X (36886) 

2 songs are from the album Pure Moods (1 1 3337) 
1 songs are from the album This Fire * (105661) 
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2 songs are from the album Villains * (93944) 
1 songs are from the album Big Calm (123837) 
1 songs are from the album Prison Bound (58282) 

1 songs are from the album The Earth Pressed Flat * [4/20] (143220) 

2 songs are from the album Clouds In My Coffee 1965-1995 [Box] (85650) 
1 songs are from the album The Piccadilly Collection * (98477) 

I songs are from the album Grown Man (92012) 

21 songs (42.0%) are from the random query 
6 songs (12.0%) are from the pop query 
6 songs (12.0%) are from the djs query 
1 7 songs (34.0%) are from the rated query 

3 songs (6.0%) originated from djAlb 

I I songs (22.0%) originated from random 
3 songs (6.0%) originated from djs 

6 songs ( 1 2.0%) originated from s avg 
3 songs (6.0%) originated from artist 

7 songs (14.000000000000002%) originated from album 
17 songs (34.0%) originated from rated 

-Percentile 0%- 20%: 40 (80%) ~— ~_ ~ > - «„ ™ , 

Percentile 20% - 40%: 2 (4%) 
Percentile 40% - 60%: 2 (4%) 
Percentile 60% - 80%: 4 (8%) 
Percentile 80% - 100%: 2 (4%) 

<P> 

Item Ratings 

Artist "The Cure" (1006316) user=0(Not Set) djs*=50/l=(Not calculated) songAverage=0/0=(Not calculated) 
songAvgScore=0.0 

Artist "Liz Phair" (1020993) user=30 djs=70/l=70 songAverage=0/0=(Not calculated) songAvgScore=0.0 
Artist "Freaky Chakra" (1009573) user=0(Not Set) djs=0/0=(Not calculated) songAverage=0/l =0 
songAvgScore=39.0 

Artist "Duncan Sheik" (1024246) user=0(Not Set) djs=0/0=(Not calculated) songAverage=80/l=80 
songAvgScore=59.0 

Artist "Tom Petty" (1020983) user=73 djs=20/l=20 songAverage=554/8=(Not calculated) songAvgScore=0.0 
Album "Great Divide" (94571) user=0(Not Set) djs=70/l=(Not calculated) songAverage=0/0=(Not calculated) 
songAvgScore=0.0 

Album "Devil Without A Cause *" (127191) user=20 djs=0/0=(Not calculated) songAverage=0/0=(Not calculated) 
songAvgScore=0.0 

«entries omitted» . 

Artist "Iron City Houserockers" (1012883) user=0(Not Set) djs=0/0=(Not calculated) songAverage=0/2=0 
songAvgScore=26.0 

Album "Superunknown" (58747) user=0(Not Set) djs=70/l=70 songAverage=0/0=(Not calculated) 
songAvgScore=0.0 

Artist "To Rococo Rot" (1032453) user=0 djs=0/0=(Not calculated) song A verage=0/0=(Not calculated) 
songAvgScore=0.0 

Album "(Not available)" (132141) user=0(Not Set) djs=80/l=(Not calculated) songAverage=0/0==(Not calculated) 
songAvgScore=0.0 

Album "Buckcherry" (143554) user=0(Not Set) djs=50/l=50 songAverage=0/0=(Not calculated) songAvgScore=0.0 
Artist "Jamie Blake" (1030814) user=0(Not Set) djs=60/l=60 songAverage=0/0=(Not calculated) 
songAvgScore=0.0 

Album "(Not available)" (45683) user=90 djs=0/0=(Not calculated) songAverage=0/0=(Not calculated) 
songAvgScore=0.0 

Album "(Not available)" (45676) user=90 djs=0/0=(Not calculated) songAverage=0/0=(Not calculated) 
songAvgScore=0.0 

Artist "LNXS" (1012859) user=0(Not Set) djs=70/l=70 songAverage=69/2=35 songAvgScore=43.5 
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Artist "Kenny Wayne Shepherd" (1024272) user=0(Not Set) djs=0/0=(Not calculated) songAverage=0/I=(Not 

calculated) songAvgScore=0.0 
Album "The Ghost Of Tom Joad" (89708) user=0(Not Set) djs=0/l=0 songAverage=0/0=(Not calculated) 

songAvgScore=0.0 

Artist "(Not available)" (1001434) user=0(Not Set) djs=10/l=(Not calculated) songAverage=0/0=(Not calculated) 

songAvgScore=0.0 
Explicitly Rated Songs 



10 
11 

12 

13 

14 

15 
16 

17 
18 



songID 



372519 



385517 



673074 



query 

comm 

rated 

52/0 

rated 

52/0 



origin status 
album! DartisID 



rated 
46/0 
rated 
49/0 



(14,28,77,) 
rated rated 
52/0 51/0 



P 

41860 
P 

43305 
P 

74082 



ord score lastP. bds 
artist title album 
5 79 100/30 0/0 

1015576The Lightning Seeds 
9 79 100/30 0/0 



impl. rating(t) djs netP. 

49 70/49(1) 52/0 

Sense Sense (14,77,) 

49 70/49(1) 52/0 



1016600Madortna 



Cherish The Immaculate Collection 



Fields Of Gold: The Best Of Sting 1984-1994 



14 79 100/30 0/0 49 70/49(1) 52/0 
1025801 Sting We'll Be Together - (previously unre leased version) 



673069 rated rated P 

. 52/0 . 44/0 74082 . 
The Best Of Sting 1984-1994 
123761 rated rated P 

52/0 48/0 14393 

23, 77, ) 

1388605rated rated P 

52/0 55/0 148836 

(14,77,) 
1062093 rated rated P 

52/0 57/0 118154 
7th Time* (14,77,) 
867248 rated rated P 

52/0 40/0 95006 
In Hi-Fi (14, 77,) 
227025 rated rated P 

52/0 48/0 25421 

(13, 14, 16,24,77,) 
857682 rated rated P 

52/0 50/0 93944 

108 1855 rated rated N 

52/0 3 8/0 119843 



18 79 
1025801 Sting 
04,77,) 

22 79 100/30 
1000398 Bryan Adams 



(14, 77,) 

100/30 0/0 49 70/49(1) 52/0 
They Dance Alone (Cueca Solo) Fields Of Gold: . 



0/0 49 
Summer Of '69 



70/49(1) 

So Far So Good 



52/0 
(13, 14, 



19 79 100/30 0/0 49 
1015285Ben Lee Cigarettes Will Kill You 



70/49(1) 52/0 
Breathing Tornados * 



29 79 100/30 
1 005865 Cornershop 



0/0 



49 



70/49(1) 



52/0 



Brimful Of Asha When I Was Born For The 



16 79 
1021 869 R.E.M. 



100/30 0/0 49 
Wake-Up Bomb, The ■ 



70/49(1) 52/0 
(live) New Adventures 



42 79 100/30 0/0 
1006149Crosby, Stills & Nash 

44 79 100/30 0/0 
1028256 The Verve Pipe Cattle 
-1 79 100/30 0/0 



1024639 Sixpence None The Richer 



49 70/49(1) 52/0 
Southern Cross CSN [Box] 

49 70/49(1) 52/0 
Villains* (14,78.) 
49 70/49(1) 52/0 



We Have Forgotten 



Sixpence None The Richer * 



(14,77,) 



454986 rated 
52/0 

Singles... 
455822 rated 



rated N 
46/0 50795 
(14,77,) 



664522 



rated 
52/0 42/0 
(14,77,) 



) 

990161 



544499 



) 

857683 



990158 



rated 
52/0 

rated 
52/0 
rated 
52/0 

rated 
52/0 
rated 
52/0 



rated 
47/0 

rated 
44/0 
rated 
47/0 

rated 
49/0 
rated 
50/0 



P 

50895 
N 

73173 
N 

1 10565 
P 

60022 
N 

93944 
N 

110565 



-1 79 100/30 
1020940Pet Shop Boys 

31 79 100/30 
1020983 Tom Petty 



0/0 
Heart 

0/0 



49 70/49(1) 52/0 
Discography - The Complete 



49 



70/49(1) 



52/0 



Face In The Crowd, A Full Moon Fever 



-I 



79 



100/30 



1016600 Madonna 



-1 79 
1027386Train 
12 79 



100/30 

Days 

100/30 



0/0 
Secret 

0/0 

Train 

0/0 



49 70/49(1) 52/0 
Bedtime Stories (7, 14, 24, 76, 77, 



49 70/49(1) 
(14, 77,) 



49 



1025830The Stone Roses Love Spreads 



70/49(1) 
Second Coming 



52/0 

52/0 
(14, 77, 



-1 79 100/30 0/0 

1 028256The Verve Pipe Veneer 

-1 79 100/30 0/0 

1027386Train Blind Train 



49 70/49(1) 52/0 
Villains* (14,78,) 
49 70/49(1) 52/0 
(14, 77,) 



WO 01/35667 



PCT/US00/30919 



18 

19 lll9487rated rated N -I 79 100/30 0/0 49 70/49(1) 52/0 

52/0 55/0 123589 1028125 Various Artists Block Rockin* Beats - The Chemical 
Brothers Digital Empire: Electronica's Best (14, 77, ) 

20 458446 rated rated P 33 79 100/30 0/0 49 70/49(1) 52/0 

52/0 37/0 51128 1 02 11 42 Pink Floyd Narrow Way Part I, The - David Gilmour 
Ummagumma (14,77,) 

«en tries omitted». 



360 


830167 


rated 


rated 


N 


-1 42 


0/0 0/0 


42 60/42(1) 


52/0 






52/0 


49/0 


90869 


1016358Lush 


Lady killers 


Lovelife* (14,77,) 


U 


songlD 


query 


origin 


status 


ord score 


lastP. bds 


imp!. rating(t) djs 


netP. 






comm 


albumlDartislD 


artist title 


album 






361 


345744 


rated 


rated 


N 


-I 42 


0/0 0/0 


42 60/42(1) 


52/0 






52/0 


49/0 


38706 


1013691 Journey Faithfully 


Time Cubed [Box] 


(14,77, 


362 


1012355rated 


rated 


N 


-1 42 


0/0 0/0 


42 60/42(1) 


52/0 






52/0 


45/0 


113423 


102363 1 Savage Garden To The Moon & Back Savage Garden 






(14,77,).. _ 












363 ~ 


673063 


rated 


rated 


N~ 


-1 42 


o/o" ~0/0~ 


42 60/42(1) 


52/0 






52/0 


47/0 


74082 


1025801 Sting 


When We Dance 


- (previously unreleased) 


Fields 



364 

365 
366 

367 

368 

369 

370 

371 

372 

373 

374 

375 



Of Gold: The Best Of Sting 1984-1994 



1383771 rated rated N 

52/0 46/0 148392 

(14,77,) 

499807 rated rated N 

52/0 51/0 55333 

1078501 rated rated N 

52/0 35/0 119582 

(14, 77,) 

1327003 rated rated N 

52/0 43/0 142307 
Henriksen 
1212748rated 

52/0 
[Limited] 
345875 rated 

52/0 

(14, 77,) 

1233646 rated random N 

52/0 40/0 134584 

Time... [ECD] (14,77,) 

573363 rated random N 

52/0 40/0 63494 



(14,77,) 
-1 42 0/0 
102 1623 The Prodigy 



0/0 42 60/42(1) 52/0 
Smack My Bitch Up Fat Of The Land 



-1 42 0/0 0/0 42 60/42(1) 52/0 
1 023239 Rush Tom Sawyer Chronicles ( 1 4, 77, ) 
-1 42 0/0 0/0 42 60/42(1) 52/0 
1015272Led Zeppelin Thank You - (stereo) BBC Sessions * 



-I 41 0/0 0/0 
1 0394 72 Tommy Henriksen 



(14,77,) 
rated N 
63/0 132410 
(14,77,) 
random N 



41 59/41 (1) 
Dreaming In Colors 



-1 40 
1027798U2 



0/0 



0/0 



40 



•1 



37 



All I Want Is You 



100/30 0/0 



52/0 
Tommy 

57/40(1) 52/0 
The Best Of 1980-1990 



10/07(1) 



52/0 



36/0 38717 101 3699Joy Of Cooking Three Day Loser American Originals 



-1 37 100/30 
103773 1 Britney Spears 

-1 37 100/30 
1027743 Twisted Sister 



Nasty Cuts-Best Of Twisted Sister (15, 16, ) 



339153 rated random N 

52/0 41/0 37973 
Up (14,77,) 



I233649rated 
52/0 

Time... [ECD] 

141 1604 rated 

52/0 43/0 

Pastels... 1983- 1985 

870674 rated random 
52/0 43/0 
Evil Empire * 



random N 
40/0 134584 
(14,77,) 
random N 



-1 37 100/30 
1013350JethroTuII 

-1 37 100/30 
1037731 Britney Spears 

-1 37 100/30 
I020680The Pastels 



0/0 7 10/07(1) 52/0 

Crazy, (You Drive Me) Baby One More 

0/0 7 10/07(1) 52/0 

We're Not Gonna Take It Big Hits And 

0/0 7 10/07(1) 52/0 

Jeffrey Goes To Leicester Square Stand 

0/0 7 10/07(1) 52/0 

Bom To Make You HappyBaby One More 



0/0 7 
Baby Honey 



50365 
(14, 77,) 

N -I 37 100/30 0/0 7 
95367 I02l928Rage Against The Machine 
(14,77,) 



10/07(1) 
Suck On The 

10/07(1) 



52/0 



52/0 



Year Of Tha Boomerang 



WO 01/35667 



PCT/US00/30919 



19 



376 


1233647rated 


random 


N 


-I 36 100/30 


0/0 6 09/06(1) 52/0 




52/0 


23/0 


134584 


1037731 Britney Spears 


Sometimes Baby One More Time... 




[ECD] (14,77.) 








377 


990162 rated 


rated 


N 


-1 35 0/0 


0/0 35 50/35(1) 52/0 




52/0 


39/0 


110565 


1027386Train Rat 


Train (14,77.) 


378 


578086 rated 


rated 


N 


-1 35 0/0 


0/0 35 50/35(1) 52/0 




52/0 


49/0 


64109 


1028073 Van Halen 


Top Of The World For Unlawful 




Carnal Knowledge 


(14,77.) 


379 


948179 rated 


rated 


N 


-1 35 0/0 


0/0 35 50/35(1) 52/0 




52/0 


50/0 


104678 


10 15374 Lemonheads 


6ix Car Button Cloth ( 1 4, 77, ) 


380 


870670 rated 


rated 


N 


-1 35 0/0 


0/0 35 50/35(1) 52/0 




52/0 


42/0 


95367 


102 1928 Rage Against The Machine Down Rodeo Evil 




Empire * 


(14,77.) 






381 


1327649rated 


rated 


N 


-1 35 0/0 


0/0 35 50/35(1) 52/0 




52/0 


55/0 


142358 


1 003 125 Blur 1992 


13 [Limited Edition] * (14, 77, ) 


382 


11 64473 rated 


random N 


-1 33 100/30 


0/0 3 04/03 (1) 52/0 




52/0 


40/0 


127996 1017147JohnMartyn 


Glory Box The Church With One 




Bell* (11,) 










383 


I004142rated 


rated 


N 


•1 31 0/0 


0/0 31 44/31 (1) 52/0 




52/0 


50/0 


112437 


1 020 1 56 Original Soundtrack Da Funk - Daft Punk The 




Saint (6, ) 










.~J84 


„J 00594 1 rated . 


_ .rated _ 


.n;..„ 


A 28__0/0 


.,0/0 .„28 . J 40/28(1) 52/0. _ 




52/0 


29/0 


112611 


101 1710Heart StrandedThese Dreams - Heart's Greatest Hits * 




(14,77.) 








385 


53 1917 rated 


rated 


N 


-1 28 0/0 


0/0 28 40/28(1) 52/0 




52/0 


48/0 


58747 


102521 3 Soundgarden 


Fell On Black Days Superunknown 




(14,77,) 








386 


224547 rated 


rated 


N 


-1 25 0/0 


0/0 25 36/25(1) 52/0 




52/0 


45/0 


25172 


1006025 Crash Test Dummies Untitled God Shuffled His Feet 




(14,77,) 








387 


991308 rated 


random N 


-1 21 0/0 


0/0 21 30/21 (1) 52/0 




52/0 


41/0 


1 1 0722 1009352Foo Fighters 


New Way Home The Colour & The Shape 




(14,78.) 








388 


531918 rated 


random N 


•1 14 0/0 


0/0 14 20/14(1) 52/0 




52/0 


44/0 


58747 


10252 13 Soundgarden 


Mailman Superunknown (14, 77, 



) 



Implicitly Rated Songs 



# songlD query origin status 
comm albumlDartisID 

1 559756 random album P 

52/0 40/2 61860 
Giants (14.77,) 

2 857672 random djAlb P 

52/0 36/2 93944 

) 



ord score lastP. bds impl. rating(t) djs netP; 
artist title album 
6 65 100/20 0/0 
l026856They Might Be Giants 



45 95/43(2) 1 0/1 

Toddler Hi way They Might Be 



2 63 100/20 0/0 43 81/36(2) 90/5 

1028256The Verve Pipe Drive You Mild Villains* (14,78. 



1212736djs album P 

52/0 53/3 132410 

(K77,) 
12 12744 random album R 

52/0 61/3 132410 

1980-1990 [Limited] (14, 77, 
778854 random album R 

52/0 46/2 85650 
In My Coffee 1965-1995 [Box] 

778868 random album P 

52/0 46/2 85650 

Coffee 1965-1995 [Box] (14, 77, ) 



10 61 100/20 0/0 41 80/36(2) 50/3 

1 027798 U2 New Year's Day The Best Of 1 980- 1 990 [Limited] 

-1 61 100/20 0/0 41 80/36(2) 40/2 

1027798U2 Sweetest Thing - (The Single mix) The Best Of 



) 

-1 61 100/20 0/0 41 80/36(2) 52/3 

1 024526Carly Simon Do The Walls Come Down Clouds 
(14,77,) 

8 61 100/20 0/0 41 80/36(2) 52/3 

1024526Carly Simon Better Not Tell Her Clouds In My 



41 



-! 61 100/20 0/0 41 

1024526Carly Simon Play With Me 

-1 61 100/20 0/0 
1024526Carly Simon 
Clouds In My Coffee 1965-1995 [Box] 

-1 61 100/20 0/0 41 

1024526Carly Simon Danny Boy 



WO 01/35667 

7 1089955 random album R 

52/0 45/2 120604 
Do... (14,77,) 

8 1 089962 random album R 

52/0 45/2 120604 
Do... (14,77,) 

9 385512 random album R 

52/0 47/2 43305 
Collection (14,28,77,) 

10 778844 random album R 

52/0 42/2 85650 
1965-1995 [Box] (14, 77,) 

11 778877 random album R 

52/0 42/2 85650 
urtreleased) 

12 778855 random album R 

52/0 40/2 85650 
1965-1995 [Box] (14, 77,) 

13 12 12734 random album R 

52/0 41/2 132410 
Best Of 1 980- 1 990 [Limited] 

14 ..,.,778848 random album 

52/0 37/2. 
Coffee 1965-1995 [Box] 

15 385563 djs artist 

52/0 49/2 

16 778847 random album 

52/0 37/2 
Coffee 1965-1995 [Box] 

17 778894 random album 

52/0 37/2 
Coffee 1965-1995 [Box] 

18 778890 random album 

52/0 37/2 
[Box] (14,77,) 

19 778856 random album 

52/0 37/2 
1965-1995 [Box] (14, 77,) 

20 1212752djs album R -1 60 

52/0 48/2 132410 1027798 U2 
[Limited] (14,77,) 



20 

-1 61 100/20 0/0 41 
10177l6John Mellencampl Need A Lover 

-1 61 100/20 0/0 41 
101 771 6 John Mellencamp Authority Song 



PCT/US00/30919 

80/36 (2) 52/3 
The Best That I Could 

80/36 (2) 52/3 
The Best That I Could 



-I 61 100/20 0/0 41 80/36(2) 50/3 
1 0 1 6600 Madonna Papa Don't Preach The immaculate 



80/36 (2) 52/3 
Clouds In My Coffee 



80/36 (2) 



52/3 



Angel From Montgomery - (prev. 
(14, 77,) 



80/36 (2) 52/3 
Clouds In My Coffee 



100/20 0/0 



JOO/20 0/0 



-1 61 
1027798U2 
(H, 77,) 
.R . .-1 60 
85650 1024526CarIy Simon 
(14,77,) 

P 38 60 100/20 
43307 10l6600Madonna 

R -1 60 100/20 
85650 1024526CarIy Simon 
(14,77,) 

R -1 60 100/20 
85650 1024526Carly Simon 
(14,77,) 
R -1 60 100/20 
85650 1024526Carly Simon 



41 



80/36 (2) 



Trash, Trampoline And The Party Girl 



.40. 



80/36(2) ... 



50/3 
The 

.52/3 _w 



Julie Through The Glass Clouds In My 

0/0 40 80/32(3) 60/6 
Erotica Erotica (14,77,) 
0/0 40 80/36(2) 52/3 
Boys In The Trees Clouds In My 

0/0 40 80/36(2) 52/3 
Nobody Does It Better Clouds In My 



R 
85650 



-I 60 100/20 
1 024526 Carly Simon 



0/0 
Why 

0/0 40 
Dink's Blues 



100/20 0/0 40 
Love Comes Tumbling 



40 80/36(2) 52/3 
Clouds In My Coffee 1965-1995 



80/36(2) 52/3 
Clouds In My Coffee 

80/36(2) 40/2 
The Best Of 1980-1990 



«en tries omitted». 



2314 



2315 



2316 



2317 



2318 



141 1055 random random N -I 23 100/20 0/0 3 
52/0 50/3 111845 l026459TalI Dwarfs Crocodile 

) 



434293 pop 
52/0 

) 

615943 pop 
52/0 
Darkness 
141 1059djs 
52/0 

) 

1411054djs 
52/0 



djArt 
52/3 



N 



-1 



22 



0/0 



0/0 



22 



00/00 (4) 
Stumpy * 

39/14 (4) 



48566 I0195I2Nine Inch Nails Ruiner The Downward Spiral 



-I 22 0/0 
1022782 Tom Robinson 



djArt N 
52/3 68246 
(14,77,) 

random N -I 22 100/20 0/0 2 00/00(4) 
42/2 11 1845 I026459TaIl Dwarfs Jesus the Beast Stumpy * 



0/0 

(14,77, 

40/6 
(14,77, 



0/0 22 39/14(4) 40/6 
Winter Of 79, The Power In The 



0/0 

(14,77, 



random N -I 22 100/20 0/0 2 00/00(4) 0/0 

40/2 1 1 1 845 1 026459Tall Dwarfs The Severed Head of Julio Stumpy * 



(14,77.) 



WO 01/35667 



2319 I411069random random N 



PCT/US00/30919 



2320 
2321 



2324 
2325 

2326 

2327 

2328 

2329 
2330 



-I 



21 

22 



100/20 0/0 



52/0 40/2 



111845 1 026459Tal! Dwarfs 



Dessicated 



14ll070djs 
52/0 

) 



random N -I 22 100/20 0/0 2 
40/2 1 11 845 1 026459Tall Dwarfs Two Minds 



00/00 (4) 
Stumpy * 

00/00(4) 
Stumpy * 



0/0 

(14, 77, 
0/0 

(14, 77, 



songID 



931183 



query 
comm 
djs 
52/0 



origin status 
albumlDartisID 
N 



s avg 
37/2 



) 



2322 560002 random random 

52/0 47/2 
(14, 16,77,) 

2323 1125549random artist 

52/0 40/2 



) 

328929 random savg 
52/0 41/2 

1073535djs savg 
52/0 ... 46/2 
- 04,77,) 

1 064098 random djs 
52/0 

Glide In Blue 
651483 random 

52/0 
Ode... [Box] 
829989 random savg 

52/0 46/2 

) 

553197 



102305 
N 

61888 
N 

124176 
N 

36886 
N 

119192 
N 

118335 



ord score lastP. bds 
artist title album 
-1 19 0/0 0/0 19 

1012081 Robyn Hitchcock Yip Song, The 



impl. rating(t) djs netP. 



39/14(4) 
Greatest Hits 



25/4 
(14,77, 



-I 19 0/0 

!026872Thin Lizzy 

-1 19 0/0 

1 023542 Santana Bella 



0/0 



19 



Killer On The Loose 



26/09(4) 



52/8 



Life Live 



0/0 19 40/16(3) 10/1 
Best Of Santana (Legacy) * (14, 77, 



-1 19 0/0 0/0 19 
1012859 INXS Faith In Each Other 
-1 18 0/0 0/0 18 
1 02 1 1 86The Pixies — Gouge Away 



43/15(4) 10/2 
X (14,77,) 
46/16(4) 0/0 
. Death To The Pixies -~ 



651476 



djs 
52/0 
djs 
52/0 



52/3 

(14,43.) 
s avg N 
47/2 72015 
(14,77,) 
N 

90854 
N 

61087 
N 

72015 



-I 18 0/0 0/0 

1030720 Apollo Four Forty 



18 



26/09(4) 



40/6 



Ain't Talkin* 'Bout Dub Electro 



-I 18 0/0 
1014381 Carole King 



0/0 



18 



39/14(4) 



10/2 



Where You Lead A Natural Woman: The 



-1 17 0/0 0/0 
101 3280 Jefferson Airplane 



s avg 
44/2 
s avg 
41/2 



Woman: The Ode... [Box] (14, 77, ) 



-1 17 0/0 
1026455 Talk Talk 
-1 17 0/0 
1014381 Carole King 



17 39/14(4) 10/2 
Crazy Miranda Bark (14, 77, 

17 39/14(4) 10/2 
It's My Life (14,77,) 
17 39/14(4) 10/2 



0/0 
Renee 
0/0 

I Feel The Earth Move A Natural 



2331 504343 



) 



djs savg N -1 15 0/0 
52/0 34/2 55865 1023614Joe Satriani 



2332 355176 random random 

52/0 47/2 
(mono) Face To Face 

2333 1233652djs djs 

52/0 41/2 
I Baby One More Time... [ECD] 

2334 958836 random random N -1 



-1 9 0/0 
1014426The Kinks 



N 

39927 
(14,77,) 

N -1 8 0/0 
1 34584 1 03773 1 Britney Spears 
(14, 77.) 
7 0/0 



0/0 15 
Summer Song 

0/0 9 



39/14(4) 0/0 
The Extremist (14,77, 



15/05(4) 



10/2 



52/0 37/2 105851 
The Isle Of Wight Festival 1970 * 



1029091 The Who 
(14, 77,) 



Most Exclusive Residence For Sale - 

0/0 8 09/04 (2) 40/2 

I Will Still Love You - (with Don Philip) 

0/0 7 09/03(4) 10/2 

I Dont Even Know Myself Live At 



Unrated Songs 



# songID query origin status 

comm albumlDartisID 

1 1011924random djAlb 

52/0 46/5 
Moods (10,) 

2 1011928random djAlb 

52/0 41/4 
Me" - Angelo Badalamenti 



P 

113337 
P 

113337 



ord 

artist 

7 



score 

title 

54 



lastP. 
album 
100/25 



1028 125 Various Artists 

11 53 100/25 
1 028 125 Various Artists 



Pure Moods 



(10,) 



bds impl. rating(t) djs netP. 



0/0 29 52/00(0) 
Crockett's Theme - Jan Hammer 



0/0 



28 



52/00(0) 



73/24 
Pure 

73/24 



Theme From "Twin Peaks - Fire Walk With 
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423652 



10 — 



11 



12 



13 



14 



15 



16 



17 



18 



19 



20 



pop random P 
52/0 52/5 47344 
The World Fail To Fall Apart 



423656 pop random P 
52/0 52/5 47344 

Fall Apart (14,77,) 
1 193855 pop random P 

52/0 52/5 

04,77.) 
423649 random random R 

52/0 52/5 47344 
Fall Apart (14,77,) 
12595 12 random random P 

52/0 52/5 

(14, 77,) 
12595 19random random P 

52/0 52/5 137097 

(14,77,) 

423657 pop random N 
52/0 52/5 47344 

Fall Apart (14,77,) 
-958997 random random -N — ~~ 

52/0 52/5 105874 
Like a Twelve Inch ( 1 4, 77, ) 
11 93 846 pop random N 

52/0 52/5 130669 
100% Columbian * 



22 

17 47 100/25 
101 8869 Peter Murphy 
(14,77,) 

34 47 100/25 
1018869Peter Murphy 
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0/0 22 52/00(0) 52/17 
Light Pours Out Of Me, The. Should 



0/0 22 
God Sends 



52/00(0) 52/17 
Should The World Fail To 



37 47 100/25 0/0 
130669 1009757Fun LovhV Criminals 



22 



52/00 (0) 



52/17 



Big Night Out 100% Columbian 



-I 47 100/25 
1018869 Peter Murphy 

45 47 100/25 
1 37097 1 028 1 25 Various Artists 

32 47 100/25 
1028 125 Various Artists 

-1 47 100/25 
1018869Peter Murphy 



-1 



0/0 



22 



Final Solution 



52/00(0) 52/17 
Should The World Fail To 



0/0 22 52/00(0) 52/17 
Portnawack - Typhoon Made On Earth 

0/0 22 52/00(0) 52/17 
Untitled - Total Eclipse Made On Earth 

0/0 22 52/00(0) 52/17 
Blue Heart Should The World Fail To 



-47 — —100/25-0/0 — 22 —52/00(0) — - 
1028125 Various Artists Freelon - Spacetime Continuum 



11 93848 pop 

52/0 
Columbian * 
!193844pop 

52/0 

(14,77.) 
1 193 845 random random 

52/0 52/5 

(14,77.) 
923902 random random 

52/0 52/5 



random N 
52/5 130669 
(14,77,) 
random N 
52/5 130669 



N 

130669 
N 

101415 



-1 47 100/25 0/0 
1009757Fun LovhV Criminals 
(14,77,) 
-1 47 100/25 0/0 
1 009757 Fun Lovin' Criminals 

-1 47 100/25 0/0 
1009757Fun Lovin 1 Criminals 

-1 47 100/25 0/0 
1009757Fun Lovin 1 Criminals 



22 



52/00 (0) 



-52/17 * 

Werks 

52/17 



View Belongs To Everyone, The 



22 52/00(0) , 
Back On The Block 



22 



52/00(0) 



52/17 
100% 

52/17 



Up On The Hill 1 00% Columbian 

22 52/00(0) 52/17 
Love Unlimited 100% Columbian 



-I 47 100/25 
1028125Various Artists 



0/0 



22 



52/00(0) 



52/17 



Tricky Presents Grassroots [EP] (14, 77, ) 



Grass Roots - Tricky/Roberto Malary Jr. 



random N 
52/5 130669 
04,77.) 
random N 



1193854 pop 

52/0 
Columbian * 
M 93849 pop 

52/0 

(14,77,) 

1 1 93852 pop random 

52/0 52/5 
You 100% Columbian * 

806170 random random N 

52/0 52/5 88136 

Hop Test Part 2 (14,77,) 

806163 random random N 



52/5 130669 



N 

130669 



-1 47 100/25 0/0 
1009757 Fun Lovin' Criminals 

-1 47 100/25 0/0 
1009757 Fun Lovin 1 Criminals 



22 52/00(0) 
AH My Time Is Gone 



22 

10th Street 



52/00(0) 



52/17 
100% 

52/17 



100% Columbian 



22 52/00(0) 52/17 
We Are All Very Worried About 



-1 47 100/25 0/0 
1009757Fun Lovin* Criminals 
(14,77,) 

-1 47 100/25 0/0 22 52/00(0) 52/17 
1028125 Various Artists Man's World, (Ifs Not) A - Strata 3 The Trip 



52/0 
Test Part 2 



52/5 88136 
(14,77.) 



-1 47 100/25 
1 028 1 25 Various Artists 



0/0 22 52/00(0) 52/17 
Anafey - Hip Optimist The Trip Hop 



«en tries omttted». 



1304 228812 pop 
52/0 
(23.) 



random N 
52/5 25620 



-1 22 0/0 
1030126The Crystals 



0/0 22 52/00(0) 52/17 
I Wonder The Best Of The Crystals 
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1313 



1314 



1315 



1316 



1317 



1318 



1319 



1320 



23 



1305 


228814 pop 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


25620 


I030l26The Crystals 


Girls Can Tell 


The Best Of The Crystals 




(23.) 










1306 


228798 pop 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


25620 


1030126The Crystals 


Oh, Yeah, Maybe, Baby The Best Of The 




Crystals (23,) 












1307 


228810 random 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


25620 


1 030 126 The Crystals 


Heartbreaker 


The Best Of The Crystals 




(23,) 










1308 


740607 pop 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


81532 


1 00809 1 EBN Get Down Ver. 2.2 


Telecommunication 




Breakdown [ECD] 


(14,77,) 






1309 


876063 pop 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


95946 


1012421 Howie B. 


Shag Music For Babies(14, 77, ) 


1310 


914734 pop 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


100059 


1020939 Pet Fatherland Pet 


(14,77,) 


1311 


882981 pop 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


96691 


!028125Various Artists 


Million Town - 


Strange Cargo (The Kruder 




& Dorfmeister Session) 


A Journey Into Ambient Groove 3 


(14, 77,) 


1312 


I 320082 pop 


random 


N 


-1 22 0/0 


0/0 22 


52/00(0) 52/17 




52/0 


52/5 


141627 


1 039729 Papa Vegas 


Something Wrong Hello Vertigo 



-1 22 0/0 
1024664 Skeleton Key 
(14,77,) 

-1 22 0/0 
1028125Various Artists 



[4/27] -(14,77,) _ 

1242704pop random N -1 22 0/0 0/0 

52/0 52/5 135883 

(14,77.) 
942415 random random N 

52/0 52/5 103598 

Skeleton Key [EP] 
11 19500 pop random N 

52/0 52/5 123589 
Empire: Electronical Best (14, 77, ) 

528565 pop random N -I 22 0/0 0/0 

52/0 52/5 58464 1025l29Sons Of Champlin 
The Best Of The Sons Of Champlin(14, 77, ) 
528568 pop random N -1 22 0/0 0/0 
58464 1025129Sons Of Champlin 
(14,77,) 

N -1 22 0/0 0/0 
103571 1024799Sloan G Turns To D 



22 



52/00 (0) 



52/17 



1038686The Hope Blister Hanky Panky Nohow Smile's OK.. 



0/0 22 52/00(0) 52/17 
World's Most Famous Undertaker, The 



0/0 22 
Take California - 



52/00 (0) 
Propellerheads 



52/17 
Digital 



52/0 52/5 
The Sons Of Champlin 
942223 random random 

52/0 52/5 
) 

942219 random random 

52/0 52/5 

(14, 77,) 
1017638random random 

52/0 48/5 

(14,77.) 



22 52/00(0) 52/17 
Get High Capitol Gold: 

22 52/00(0) 52/17 
It's TimeCapitoi Gold: The Best Of 

22 52/00(0) 52/17 
One Chord To Another ( 1 4. 77, 



N 

103571 
N 

1 14082 



-1 22 
1024799 Sloan 

-1 22 



0/0 



0/0 



22 



Good In Everyone, The 



52/00(0) 52/17 
One Chord To Another 



0/0 



0/0 



22 52/00 (0) 



52/17 



1 004 1 59David Byrne Wicked Little Doll 



Feelings * 



1321 



1323 



songID query origin status 

comm albumJDartisID 

809747 random random N 

52/0 46/5 



1322 455363 



88473 
random random N 



52/0 40/4 50841 

Peter & Gordon (Rhino) (23, ) 

814350 random djArt N 

52/0 45/5 88938 



1324 232378 



djs 
52/0 



random N 
49/5 26074 



ord score lastP. 
artist title album 
-1 22 0/0 
1015875Loop Guru 
-1 21 0/0 
1030292 Peter & Gordon 



bds imp]. rating(t) djs 



0/0 22 
Jungle ADuniya 
0/0 21 



52/00 (0) 
(14,77.) 
52/00 (0) 



netP. 
52/17 
52/17 



I Feel Like Going Out The Best Of 



40/13 



Damned (Another...) (14, 78, ) 



-1 18 0/0 0/0 18 52/00(0) 
1 02 1 734 Pulp Death II Separations ( 1 4, 77, ) 
-1 12 0/0 0/0 12 52/00(0) 20/7 
1006547The Damned Smash It Up (Parts 1 & 2) The Best Of The 
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<REPEAT> 
<ENTRY> 

<REF HREF-'http://devweb7.Iaunch.com/servlet/gateway?u=6474126&n=0.asp M /> 
</ENTRY> 
<ENTRY> 

<REF HREF- 'http://devweb7Jaunch.com/servlet/gateway?u=6474 126&n= I .asp"/> 
</ENTRY> 
<ENTRY> 

<REF HR£F= M http://devweb7.launch.com/serv!et/gateway?u=6474126&n=2.asp"/> 
</ENTRY> 
<ENTRY> 

<REF HREF^ M http://devweb7Jaunchxom/sei^let/gateway?u==6474126&n=3.asp ,, /> 
</ENTRY> 
<ENTRY> 

<REF HREF="http://devweb7.1aunch.com/servlet/gateway?u=6474126&n=4.asp"/> 
</ENTRY> 
<ENTRY> 

<REF HREF= ,, http://devweb7.1aunch.com/servIet/gateway?u=6474 1 26&n=5.asp"^ 
</ENTRY> 
<ENTRY> 

. <REF HREF="http://devweb7Jaunchxom/serv!et/gateway?u^474126&n^.asp ,, /> 

</ENTRY>. ... ........ .. , .J . _ : ^ 

<ENTRY> 

<REF HREF-'http://devweb7 .launch.com/servlet/gateway?u=6474 1 26&n=7.asp"A> 
</ENTRY> 
<ENTRY> 

<REF HREF="http://devweb7Jaunchxom/servlet/gateway?u=6474126&n=8.asp ,, /> 
</ENTRY> 
<ENTRY> 

<REF HREF="http://devweb7. launch.com/servlet/gateway?u=6474 1 26&n=9.asp"/> 
</ENTRY> 
</REPEAT> 
</ASX> 
</XMP> 
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GETPOPULAR 54 

GETRATINGS 55 

GETRATINGSCACHEUSERS 59 

GETRATINGSCACHEUSERSINTERFACE 61 
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GETSONGINFOSERVLET 64 

GETSONGRATINGSFROMDB .. 70 

INTHASH 71 
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ITEMSPROFILE . 74 

MEDIA 76 
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RECLIST 153 

SAVECLIPS 156 

SAVEPLAYLIST 158 

SIMPLECLIP 160 

SIMPLECLIPLIST — 161 

SIMPLEPLAYLIST ~ 162 

SONG 165 

SONGDATA..... . 167 

SONGGROUP , .... ... .. 174 
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SONGINFOCACHE . , . ... 178 
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UTIL ...... ... • ..... 192 
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AlbumArtistData 

package com.launch.PlaylistGenerator; 

public class AlbumArtistData 

{ 

Item album - null; 
Item artist = null; 

boolean alreadyTriedAlbum - false; 
boolean alreadyTriedArtist = false; 

public void resetO 
{ 

album = null; 
artist = null; 

alreadyTriedAlbum = false; 
alreadyTriedArtist = false; 

} 

public Item getAIbum(ItemsProfiIe items, SongData data) 

if (alreadyTriedAlbum) 

return album; ...... ~ -., - ' : - — - :: - - - - : - - : -- •• - : - 

alreadyTriedAlbum = true; 

album = items.get(data.getAlbumIDQ); 

return album; 

} 

public Item getArtist(ItemsProflle items, SongData data) 
{ 

if (alreadyTriedArtist) 

return artist; / 

alreadyTriedArtist - true; 

artist = items.get(data.getArtistID0); 

return artist; 

AlbumArtistDatajava Page 1 of 1 11/05/99 1:32 PM 
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Album Info 

package com.launch.PIaylistGenerator; 
import java.util Vector; 
public class Aibumlnfo 
{ 

int ID; 
String title; 
Artistlnfo artist; 

Vector genres; 

public AlbumInfo(int ID) 

this.ID=ID; 



public String toStringO 

return H [albumiD=" + ID + title=" + title 

• + genres=" + genresStringO + artist^' + artisttoStringQ + T; 



public String genresStringO - - - ■ 

if (genres = null) 

return M (NONE)"; 

String result = MM ; 

for (int i = 0; i < genres.sizeO; 
{ 

result = resultxoncat(genres.elementAt(i) + "); 

} 

return "(" + result + ")";. 

} 

public int getArtistlDQ throws Exception 
{ 

if (artist = null) 

throw new Exception("artist is not set for album " + ID + M (" + title + ") "); 
return artist.ID; 

} 

public boolean inGenres(short genrelD) 

{ 

if (genres =?= null) 

return false; 

return genres.contains(new Short(genrelD)); 

public boolean inGenres(GenreList userGenres) 
{ 

if (userGenres.allGenres = true) 
return true; 

if (genres = null) 

return false; 

App. 2-5 
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Artistlnfo 

package com. launch. Play I istGenerator; 
import java.util.Hashtable; 
public class Artistlnfo 

* { 

int ID; 
String title; 
Hashtable songs; 

io public ArtistInfo(int ID) 

{ 

this.ID=ID; 

songs - new HashtableO; 

« > 

public String toStringO 
{ 

return n [artistID=" + ID + title-" + title + "]"; 

public final static boolean isVariousArtists(int itemID) 
{ 

return (itemID = Constants.ARTIST_VARIOUS_ARTISTS 

|| itemID = Constants.ARTIST^ORIGIN AL_SOUNDTRACK 
25 II itemID = Constants.ARTIST_SOUNDTRACK); 

} 

} 

Artistlnfo.java Page 1 of 1 11/05/99 1:37 PM 



App. 2-7 



WO 01/35667 PCT7US00/30919 

31 

// do it the other way, check each of the genres the song is 
// in and if it's in the user's genres 

for (int i = 0; i < genres.sizeO; 
{ 

Short genrelD = (Short) genres.elementAt(i); 

if (userGenres.exists(genreID)) 
return true; 

} 

return false; 

} 

public void addGenre(short genrelD) 
{ 

if (genres = hull) 

genres = new Vector(I,l); 

. . ,// be careful not to add duplicates — ^ ™ 

Short genre = new Short(genreID); 

if (!genres.contains(genre)) 

genres.addElement(new Short(genreID)); 

} 



} 

Albumlnfo.java Page 2 of 2 1 1/05/99 1 :27 PM 
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AverageRating 

package com.launch.PlaylistGenerator; 
public class AverageRating extends Rating 
{ 

private short count = 0; 
private int sum; 

private boolean calculated = false; 
public AverageRatingO 

{ 

superO; 

} 

public AverageRating(short defaultRattng) 
{ 

super(defaultRating); 

} 

public void add(int value) 
{ 

sum value; 
count++; 

v — — calculated - false; ~ - — — > ~<~~~^ — 

} 

public short getO 
{ 

calculateQ; 
return super.getO; 

} 

public short countO 
{ 

return count; 

} 

private void calculated 
{ 

if ((calculated) 
{ 

if(count>0) 
{ 

set(Util.average(count, sum)); 
set = true; 

} 

calculated = true; 

} 

} 

public String toStringO 
{ 

String ratingStr = "(Not calculated)"; 

if (set) ratingStr = + rating; 

return sum + 7" + count + "= M + ratingStr; 

} 

} 

AverageRatingJava Page 2 of 2 1 1/05/99 1 :27 PM 
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Bandwidth 

package com. launch. Play I istGenerator; 

public class Bandwidth 

{ 

public final static short SPEED^28 = 28; 
public final static short SPEED_56 = 56; 
public final static short SPEEDJ00 = 100; 
public final static short SPEED_I28 = 128; 
public final static short SPEED_300 = 300; 
public final static short SPEED_500 = 500; 

private boolean beenset = false; 
private short value = SPEED_28; 

public BandwidthO 
{ 

} 

public Band width(short speed) — - - v — - — — - 

{ . 
value = speed; 
beenset = true; 

} 

public Bandwidth(String speed) 
{ 

if (speed = null) 
{ 

beenset — false; 

} 

else 
{ 

if(speed.equals("28 M )) 

set(SPEED_28); 
else if (speed.equaIs("56 M )) 

set(SPEED_56); 
else if (speed.equals(" 1 00")) 

set(SPEED_100); 
else if (speed.equals("128 M )) 

set(SPEEDJ28); 
else if (speed.equals( ,, 300 ,, )) 

set(SPEED 300); 
else if (speedequals("500")) 

set(SPEED_500); 

else 
{ 

beenset = false; 

} 

} 

} 

public String toStringO 
{ 

if(value = SPEED_28) 

return M 28.8k"; 
else if (value = SPEED_56) 

return "56k"; 
else if (value = SPEED_100) 



App. 2-9 



WO 01/35667 PCT/US00/30919 

34 

return "100k"; 
else if (value = SPEEDJ28) 

return" 128k"; 
else if (value = SPEED_300) 

return "300k"; 
else if (value = SPEED_500) 

return "56k"; 
return "UNKNOWN (" + value + ")"; 

} 

public short get() 
{ 

return value; 

} 

public void set(short speed) 
{ 

if (speed = SPEED_28 

II speed == SPEED_56 
|| speed = SPEED_1 00 
|| speed = SPEED J 28 

. ... || speed SPEEDJ300-— — — — -~ ™ 

|| speed = SPEED_500) 



{ 



} 

else 



value = speed; 
beenset = true; 



beenset = false; 



public boolean Ioad(DBCorinection conn, int user ID) 
{ 



try 
{ 



userlD); 



DBResultSet rs = conn.executeSQL("exec sp_al50UserPreference_GetValue_xsxx 



if(!rs.getBOF0&& Irs.getEOFO) 
{ 

set(rs.getShort( ,, iDefauItBandwidth")); 

} 

} 

catch (DBException oops) 
{ 

UtiLdebugC'DB Exception in Bandwidth:: load: " + oops.getMessageQ); 

} 

return isSetO; 

} 

public boolean isSet() 
{ 

return beenset; 

} 

} 

Bandwidth.java Page 3 of 3 11 /05/99 1 :32 PM 
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BDSRank 

package com.launch.PlaylistGenerator; 

public class BDSRank 

{ 

short stationID; 
byte rank; 

public BDSRank(short stationID, byte rank) 
{ 

this.stationID = stationID; 
this.rank = rank; 

} 

public String toStringO 

return stationID + ":" + rank; 

} 

BDSRankjava Page 1 of 1 1 1/05/99 1:26 PM^^ — ^ „ — 
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CachedRating 

package com.launch.PlaylistGenerator; 
import java.io.*; 
import java.util.Date; 
/•* 

* This class is used to model a single rating in the cache. 
♦V 

public final class CachedRating implements Serial izable 
{ 

public int userlD; 
public int item ID; 
public byte rating; 
public byte type; 

private Date created - new DateO; 

//- 

public CachedRating(int userlD, int itemID, byte rating, byte type) 
{ 

this.userlD = userlD; 

i ji " — — this.itemID - itemJD; — ~ - — — - . '- - - 

~ this.rating — rating; 

this.type = type; 

} 

public final String toStringO 
{ 

retum( M user n + userlD + ", itemID:" + itemID + ", rating:" + rating + ", type:" + 
typeString(type) + ", date: " + created.toStringO + UtiLnewLine); 
} 

public final static String typeString(byte type) 
' { 

if (type = Constants.ITEM_TYPE_SONG) 

return "song"; 
else if (type = Constants. ITEM_TYPE - ALBUM) 

return "album"; 
else if (type = Constants . I TEM_T YPEA RTI ST) 

return "artist"; 
return "unknown"; 

} 

public String hashKeyO 
{ 

return itemID + ":" + type; 

} 

} 

CachedRatingjava Page 1 of 1 1 1/05/99 1:35 PM 
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Clip 

package com . launch . Play 1 istG enerator; 
import java.util. Date; 
public class Clip 
{ 

public final static byte TYPE_NONE = 0; 

public final static byte TYPE_NEWS = 1 ; 

public final static byte TYPE_AD = 2; 

public final static byte TYPEJNTERSTITIAL = 3; 

public final static byte TYPEJTIP » 4; 

public final static byte TYPESONG = 5; 

public final static byte TYPEJBROADCAST = 6; 

public int ID; 

public byte type; 

public int medialD; 

public Date lastPlayed; 

public String name, directory, server, filepath; 

public MediaList media; 

private boolean set = false; 
public Clip(byte type) 
{ 

this.type = type; 

media = new MediaListO; 

} 

public Clip(int ID, byte type) 
{ 

this(type); 
this.ID = ID; 

} 

public CIip(int ID, byte type, int medialD, String name, Date lastPlayed) 
{ 

this(ID, type); 
this.ID =ID; 
this.medialD = medialD; 
this.name = name; 
this.lastPlayed = lastPlayed; 

} 

public byte type() { return type; } 

public boolean isSet0 { return set; } 

private void setDirectory(String newDir) 
{ 

if (!newDir.equaIs(" ")) 
{ 

directory = newDir; 

} 

} 

public void logPlay(DBConnection conn, int userlD) 
{ 

String sql = ""; 

if (type = TYPE_SONG) 

sql = "exec sp_lcLogPlaySong_isud " + userlD + " + medialD + " + ID + 

origin; 
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else if (type = TYPE AD) 

sql = "exec spJcLogPlayAd_isud " + userlD + " + medialD + " + ID; 
else if (type — TYPENEWS) 

sql = "exec sp_IcLogPIayNews_isud " + userlD + ", " + medialD + ", B + ID; 
else if (type = TYPE_TIP) 

sql = "exec sp_IcLogPIayTip_isud " + userlD + H + medialD + ", " + ID; 
// else if (type — TYPE_BROADCAST) 

// sql = "exec sp_lcLogPlayBroadcast_isux " + userlD + H , " + mediaType; 

try 
{ 

conn.executeUpdate(sql, true); 

} 

catch (DB Exception e) 

{ 

System.err.printIn("DBException in Clip:logPlay:" + e.toStringO); 

} 

} 

public boolean getPath(DBConnection conn, ClipSchedule schedule) 

^ if (type = TYPE_NONE) 

return false; 

SimpleCIipList list = null; 

if (type — TYPE_SONG) 

list = schedu!e.playlist.songs; 
else if (type — TYPE_AD) 

list = schedule.playlist.ads; 
else if (type = TYPEJTIP) 

list - schedule.playlisttips; 
else if (type = TYPENEWS) 

list = schedule.playlist.news; 

if (list = null) 

return false; 

SimpleGlip yip = list.popO; 

if (yip — null) 

return false; 

medialD = yip.medialD; 
ID =yip.ID; 
origin =yip.origin; 



try 
{ 

DBResuItSet rs = conn.executeSQLC'exec sp_IcGetMediaPath_xsxx " + medialD); 

if (irs.getBOFO && Irs.getEOFO) 
{ 

setDirectory(rs.getString("directory")); 
server = rs.getStrtng( n server"); 
filepath = rs.getString("fiIepath"); 

set = true; 
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catch (DBException e) 
{ 

System.err.printIn( f, DBException in Clip::getPath: " + e.toStringO); 

} 

return set; 

} 

/♦ 

public boolean pop(DBConnection conn, int userlD, int context) 
{ 

set = false; 

try 

{ 

DBResultSet rs; 
String the_command; 

int contextNum = 0; 

if (context > 1) contextNum = 1 ; 

if (type=TYPE_BROADCAST) 

{ * 

- — — — the_command= ,, exec BROADCAST_SP + " M + userlD + w , " + type + 

context; 

~ . ; ~ } 

else 

{ 

String stored_proc = null; 

if (type = TYPE_AD ) stored_proc « ADS_SP; 
else if (type = TYPE_TIP ) stored _proc = TIPS_SP; 
else if (type = TYPEJSfEWS) stored jproc = NEWS_SP; 
else stored_proc - SONG_SP; 

the_command= "exec " + stored_proc + n " + userlD + n + contextNum; 

} 

rs = conn.executeSQL(the_command); 
if (!rs.getBOF0&& !rs.getEOF0) 
{ 

setDirectory(rs.getString( w directory w )); 
server = rs.getString( n server"); 
filepath = rs.getString("filepath"); 

set = true; 

} 

} 

catch (DBException e) 
{ 

System.erir.println^DBException in Clip::pop: " + e.toStringO); 

} 

return isSetO; 

} 

*/ 

public String pathO 
{ 

return server 

+ directory 
+ 7" 

+ filepath; 

} 

public String toStringO 
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{ 

return "Clip type (" + typeName() + id = " + medialD 
+ ", lastPlayed = " + lastPlayed 
-f media = " + media.toStrtng() 
+ path = " + path(); 

} 

public PlaylistEntry toPlay!istEntry(short mediaType) 
{ 

PlaylistEntry entry = new PlaylistEntryO; 
entry.medialD = media.getlD(mediaType); 
entry .title = name; 

entry, filepath = media. getFilepath(mediaType); 
return entry; 

> 

public SimpIeClip toSimpleClip( short mediaType) 

return new SimpleClip(ID, media.getlD(mediaType)); 

} 

public String typeNameO 
{ 

switch(type) 
{ 

case TYPE_AD: 

return M Ad"; 
case TYPE_BROADCAST: 

return "Broadcast"; 
case TYPEJNTERSTITIAL: 

return "Interstitial"; 
case TYPE_NEWS: 

return "News"; 
caseTYPE_TIP; 

return "Tip"; 
• case TYPESONG: 

return "Song"; 

} 

return "?"; 

} 

public String URLO 
{ 

return server 

+ directory 
+ 7" 

+ filepath; 

} 

} 

Clip Java Page 5 of 5 1 1/05/99 1:32 PM 
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ClipCollection 

package com. launch. Play listGenerator; 

import java.util.Hashtable; 

public class ClipCollection extends Hashtable 

{ 

public Clip put(int clipID, Clip aClip) 
{ 

return (Clip) put(new Integer(cliplD), aClip); 

} 

public Clip get (int clipID) 
{ 

return (Clip) get(new Integer(clipID)); 
ClipCollection.java Page I of 1 1 1/05/99 1:26 PM 
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ClipSchedule 

package com. launch. Play 1 istGenerator; 
import java.util. Date; 

import javax.servletServletOutputStream; 

public class ClipSchedule 

{ 

private Date dbDate; 

private int userlD, lastBroadcast, currentBroadcast; 

private boolean set ~ false; 
public SimpIePlaylist playlist; 

public ClipSchedule (int userlD) 

{ 

this.userlD = userlD; 

} 

public void init(DBConnection conn) 

""" set = false; ■ 

try 
{ 

DBResultSet rs = conn.executeSQL( M exec sp_lcGetClipSchedule_xsxx H + userlD); 

if (irs.getBOFO && Irs.getEOFO) 

{ 

dbDate = rs.getTimestamp( ,f dbDate"); 

lastBroadcast =rs.getInt( ,, IastBroadcastID"); 

currentBroadcast = rs.getInt( H broadcastip H ); 

playlist = SimpIePIaylist.f>omBytes(rs.getBytes("playlist ,, )); 

} 

else 
{ 

dbDate = new DateO; 

} 

// the first time a playlist is created for a user, the dates will be null 

if (playlist != null) 
{ 

if (playlistlastAd = null) playlistlastAd = dbDate; 
if(playlist.lastNews — null) playlist. lastNews = dbDate; 
if (playlist.IastTip = null) playlist.lastTip = dbDate; 
set = true; 

} 

} 

catch (DBException e) 
{ 

System.err.println( M DBException in ClipSchedule:: init:" + e.toStringO); 

} 

} 

private long dateDiff(Date diffMe) 
{ 

if(diffMe = nuil) 

diflMe = new Date(0); 
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return (long) ((dbDate.getTimeO - difTMe.getTimeO) / (1000.0 * 60)); 

} 

public byte nextClipType(boolean debug, ServletOutputStream out) 
{ 

long adDiff, newsDiff, tipDiff; 

while (true) 
{ 

adDiff =dateDiff(playlistlastAd); 
newsDiff = dateDiff(pIaylist.lastNews); 
tipDiff = dateDiff(playlist.!astTip); 



if(debug) 
{ 



minutes"); 



newsDiff) + " minutes"); 



minutes"); 

} 



UtiI.out(out, "dbDate is " + dbDate.toStringO); 

Util.out(out, "lastAdDate is " + playlistlastAd); 

UtiLout(out, "next ad in " + (Constants. AD_THRESHOLD - adDiff) + " 



Util.out(out, "lastNewsDate is "+ playlistlastNews); 

Util.out(out, "next news clip in " + (Constants.N E WSJTHRESHOLD - 



Util,out(out, "IastTipDate is " + playlistlastTip); 

Util.out(out, "next tip in " + (Constants.TIPJTHRESHOLD - tipDiff) + " 



if (play list = null) 
{ 

System.err.println(new DateO.toStringO + n nextClipType: userlD " + userlD + 

1 has no/invalid playlist"); 

return Clip.TYPE_NONE; 

} 

if (currentBroadcast > lastBroadcast) 
{ 

if (debug) Utihout(out, "getting broadcast"); 
lastBroadcast - currentBroadcast; 
return ClipTYPE_BROADCAST; 

> 

else if (adDiff >= Constants. AD_THRESHOLD) 
{ 

if (debug) Util.out(out, "playing AD"); 
playlist last Ad - dbDate; 



out of ads"); 



if (playlist.ads.isEmptyO) 

System.err.println(new DateO.toStringO + M userlD H + userlD + " is 

else 

return CIipTYPE_AD; 

} 

else if (newsDiff >= Constants.NE WSJTHRESHOLD) 
{ 

if (debug) Util.out(out, "playing NEWS"); 
playlistlastNews = dbDate; 

if (playlist.news.isEmptyO) 

System.err.println(new DateO.toStringO + " userlD " + userlD + " is 

App. 2-19 



WO 01/35667 
out of news"); 



out of tips"); 



out of songs"); 



PCT/US00/30919 



44 



else 



return CIip.TYPE_NEWS; 



} 

else if (tipDiff >= Con stants .TIP TH RES HOLD) 
{ 

if (debug) Util.out(out, "playing TIP"); 
playlist.lastTip - dbDate; 

if (playlisttips.isEmptyO) 

System.en-.printin(new DateO-toStrihgO + " userlD " + userlD + " is 



} 

else 
{ 



else 



return Clip.TYPE_TlP; 



if (debug) Util.out(out, "playing SONG"); 
if (playlistsongsisEmptyO) 



} 

else 



System.err.println(new DateO-toStringO + " userlD " + userlD + "is 
return Clip.TYPE_NONE; 

return Clip.TYPE_SONG; 



} 



} 

//return Clip.TYPE_NONE; 



} 

ClipSchedulejava 
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Constants 

package com.launch.PlaylistGenerator; 

public interface Constants 

{ 

// live 



public final static String DBSOURCE 
public final static String DBJJSERNAME 
public final static String DB_PASSWORD 
public final static String DB_DBNAME 
public final static String DB_SERVER 
public final static short DB_PORT 



= "LAUNCHcast"; 
=» "dbClient"; 
= "83kareem23"; 
= "dbLaunchProd"; 
= "209.67. 158.1 9 tt ; // DB3 
= 1433; 



public final static String STREAMJJRL = M http://lcplaylist. launch.com/servlet/gateway"; 
public final static String STREAM_SERVER = ,, http://lcstream.launch.com"; 
*/ 

// development 

public final static String DB_SOURCE = "LAUNCHcast"; 

public final static String DBJJSERNAME = "dbClient"; 

public final static String DB_PASS WORD = "29Idiocy99"; 

public final static String DBJDBNAME = "dbLaunchProd"; ;,___ M _ J ^_ 

public final static String DB_SERVER "~ = "zeus"; 
public final static short DB_PORT =1433; 

public final static String STREAM_URL - "http://devweb7. launch.com/servlet/gateway"; 
public final static String STREAM_SERVER = "http://devweb7.Iaunch.eom/F"; 
public final static int RIAA MAX SONGS FROM ALBUM = 2; 
public final static int RI A A_M AX_SONG S_BY_ARTIST = 3; 
public final static int BDS_SCORE_MAX_POINTS =41; 
public final static int BDS_SCOREJPOINTBAR = 20; 

public final static int DEFAULT_LASTPLAYED_SCORE = 1 00; 

public final static int DEFAULT_MEDIATYPE 
public final static int DEFAULT_UNRATED_RATIO = 50; 
■ public final static int DEFAU LT_PICK_F ACTOR = 7; 

public final static int DEFAULTBDSJSCORE 

public final static int MAX_PERCENT__RATED_SONGS_TO__PICK = 20; 
public final static int NEW_USER_UNRATED_RATIO = 90; 
public final static int MIN_RATINGS_TO_HONOR_RATIO = 1 00; 
public final static int MIN_SIZE_FOR_NO_UNRATED = 200; 

public final static int MAX_0RD1NAL = 1000; 

// for calculating implicit based on other song ratings 

public final static int MAX_SONGS__BY_ARTIST = 4; 

// random picking 

public final static int RANDOM_SONGS_COUNT = 5000; 

// this is a percent of the total number of songs in the database 

public final static int MIN_SONGS_IN_GENRES_TO_GET_RANDOM = 5; 



= 211; // 16 Mono 



0; 



so all songs 



public final static int MIN_RATING_FOR_RATED_SOURCE 
// songs with average rating above this are considered popular 
// also change this at the top of LAUNCHCast/pIayer/getsonginfo 
public final static int POPULARJTH RESHOLD 
public final static int DEFAULT.RATTNG 



= 35; 



60 



public final static int DEFAULT_DJS_SCORE 

public final static int DEFAULT_NETP_SCORE 

public final static byte DEFAULTCOMMRATING 

public final static int MAX_RATINGS_TO_GET 

public final static int MAX_DJ_RATINGS_TO_GET 

public final static int ARTIST_VARIOUS_ARTISTS = 1028125; 

public final static int ARTIST_ORIGrNAL_SOUNDTRACK 

public final static int ARTIST_SOUNDTRACK 

public final static int DEFAULT_PLAYLIST_SIZE = 50; 

public final static int MAX^NEWSJTEMS 



= 500; 



= 58; 

= 52; // global average for 



= DEFAULT_RATJNG 
= DEFAULT_RATING; 
= DEFAULT_RATING 
= 500; 



= 1020156; 

= 1036715; 

= 0; 
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public final static int MAX_ADS = 20; 

public final static int MAXJTIPSJTEMS = 0; 

public final static int REFRESH_AT_SONGS_LEFT = 8; 

public final static int REFRESH_AT_NEW_RATINGS_COUNT = 1 5; 

public final static int ADJTHRESHOLD = 30; 

public final static int NEWS_THRESHOLD = 99999999; 

public final static int TIP_THRESHOLD = 99999999; 

public final static byte ITEM TYPE_SONG = 1 ; 

public final static byte I TEM_TYPE_ALB UM - 2; 

public final static byte ITEM_TYPE_ARTIST = 3; 

// the size of the ratings cache FOR EACH user 

public final static int RATINGS_CACHE JNITIAL_SIZE « 2000; 

public final static int RATING_UPDATE_L!ST_fNITIAL_SIZE = 100; 

// for updating the ratings caches 

public static final int PROPAG ATE_DI RTY_RATING_SLEEP_TI ME = 60 * 1000; // every 60 seconds 
public static final String POST_HEADER = "POST /servlet/playlist HTTP/1.0"; 
public static final int PORT_NUMBER =80; 

} 

Constants.java Page 2 of 2 11/05/99 1 :24 PM 
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DBConnection 

package com.launch.PIaylistGenerator; 
import java.util. Properties; 
import com.inettds.TdsDriver; 
import java,sqI.SQLException ; 
import java.sql. Statement; 
import Java. sql.Connection; 
import java.sql. Driver; 
import java.sql.DriverManager; 
import java.util. Date; 
public class DBConnection 

{ 

private Connection conn; 

public static Driver DB Driver; 

public DBConnectionO throws DB Exception 
{ 



... » - if(DBConnection.DBDriver== null) „ — 

DBConnection. in itializeDriverO; 

if (DBConnection.DBDriver — null) 
return; 

String 'Uri = "jdbc:inetdae: H 

+ Constants.DB_SERVER 

+ Constants.DB_PORT 
+ "?sql7=true&database=" 
+ Constants.DB_DBNAME 
+ "&user=" 

+ Constants.DBUSERNAME 

+ M &password=" 

+ Constants.DB_PASS WORD 
• mi. 
> 

try 
{ 

conn = DBConnection.DBDriver.connect(url, null); 

} 

catch (SQLException oops) 
{ 

throw new DBException(oops); 

} 

catch (Exception err) 
{ 

Util.debugCException: " + err.toStringO); 

} 

} 

private static void initializeDriverO 
{ 

DBDriver = new com.inet.tds.TdsDriverO; 

} 

private DBResultSet execute(String sql, boolean printSQL) throws DBException 
{ 

if (printSQL) 
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Util.debug(Util.newLine + ThreadxurrentThreadO-getNameO + " Running SQL: " + sql); 
DBResultSet myRs « new DBResultSetO; 
try 

{ 

63 

// if we don't have a query, don't run it. It'll hang 
if (sql.length() <= 0) 
return myRs; 

to Statement query = conn.createStatementO; 

if (query .execute(sql)) 
{ 

myRs.setResuitSet(query.getResultSetO); 

* } 
} 

catch (SQLException oops) 
{ 

System.err.print!n(Util.newLine + (new DateO)toStringO + n DBException: " + 
w Thread.currentThread0.getNameO + " Running; SQL: w + sql + exception: " + oopsitoStringO); 

oops.printStackTraceO; 

— — — - — — -.throw new DBException(oops);~~ ^ „ „ , , - , ., . > , t 

; _ } • , ... v ■■ _„»^ M - . - :: . . 

85 return myRs; 

} 

public void executeUpdate(String sql, boolean printSQL) throws DBException 
{ 

» if (printSQL) 

Util.debug(Util.newLine + Thread.currentThreadO.getNameO + H Running SQL: " + sql); 

try 
{ 

95 // if we don't have a query, don't run it. It'll hang 

if(sql.length0<=0) 
return; 

Statement query = conn.createStatementO; 

100 

query.executeUpdate(sql); 

} 

catch (SQLException oops) 
{ 

ids // when we call a stored proc that gets a text pointer this happens, 

//so ignore it 

if (oops.getMessage0.indexOf("Unknown datatype") > -1) 
{ 

// System.err.printIn("ignoring unknown datatype exception"); 

uo return; 

} 

System.err.println(UtiLnewLine + (new Date()).toStringO + M DBException; " + 
Thread.currentThread0.getNameO + " Running SQL: " + sql + " exception: " + oops.toStringO); 
us oops.printStackTraceO; 

throw new DBException(oops); 

} 

} 

120 

public DBResultSet executeSQL(String sql) throws DBException 
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return execute(sql, true); 



public OBResultSet executeSQL(String sql, boolean printSQL) throws DBException 
return execute(sql, printSQL); 

public DBPreparedStatement prepareStatement(String sql) throws DBException 



try 
{ 

} 



return new DBPreparedStatement(conn.prepareStatement(sql)); 



catch (SQLException oops) 
{ 

System.err.println(UtiI.newLine + (new DateO) toStringO ■+ " DBException in 
prepareStatement: M + ThreadxurrentThread0.getNarne() + " t exception: n + oops.toStringO); 
oops.printStackTraceQ; 
-~ throw new DB Exception(oops); . - . ~ w 



} 



} 



public boolean closeO throws DBException 
{ 

if (conn = null) 

return false; 



} 



try 
{ 



} 



conn.closeO; 
conn— null; 
return true; 



catch (SQLException oops) 

{ 

throw new DBException(oops); 

} 



public void finalizeO throws DBException 
{ 

// in case someone forgets 
closeO; 

} 

} 

DBConnection.java Page 4 of 4 1 1/05/99 1 :37 PM 
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DBException 

package com. launch.PlaylistGenerator; 

import java.sql.SQLException; 

public class DBException extends Exception 

{ 

SQLException oops; 

public DBException(SQLException oops) 

{ 

this.oops - oops; 

} 

public String getMessageO 
{ 

return oops.toStringO; 

} 

} 

DBException.java Page I of 1 11/05/99 1 :26 PM 
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DBPreparedStatement 

package com.launch.PlaylistGenerator; 

import java.sql.PreparedStatement; 

import java.sql.SQLException; 

import ja va.uti 1 . Date; 

public class DBPreparedStatement 

{ 

PreparedStatement statement; 

public DBPreparedStatement(PreparedStatement statement) 
{ 

this.statement = statement; 

} 

public void setBytes(int parameterlndex, byte xfl) throws DBException 

{ " 
try 

{ 

if (statement != null) 

statement.setBytes(parameterIndex, x); 

} " • ... ..... - - - ^ ^ 

} 

catch (SQLException e) 
{ 

throw new DBException(e); 

public void executeUpdateO throws DBException 
{ 

Util.debug(Util.newLine + Thread.currentThreadQ.getNameO + " Running prepared statement"); 

if (statement = null) 
return; 

try 

{ 

statement.executeUpdateO; 

} 

catch (SQLException oops) 
{ 

System.err.pririt1n(Util.newLine + (new Date()).toStringO + " DBException: + 
ThreadcurrentThreadO-getNameO + " Running Statement, exception: " + oops.toStringO); 
oops.printStackTraceO; 
throw new DBException(oops); 

} 

} 

} 

DBPreparedStatementjava Page 1 of 1 11/05/99 1:32 PM 
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DBResultSet 

package com.launch.PlaylistGenerator; 

import java.util. Date; 

import java.sql.Resu!tSet; 

import java.sqLSQLException; 

import java.sql.Timestamp; 

import java.io.lnputStream; 

public class DBResultSet 

{ 

private ResultSet rs; 

private boolean atEOF = false; 

private boolean atBOF = true; 

public void setResultSet(ResultSet aRS) throws DBException 
{ 

try 
{ 

rs = aRS; 
if(rs!=null) 

atBOF = Irs.nextQ; 



catch (SQLException oops) 

throw new DBException(oops); 

} 

} 

public int getInt(String columnName) throws DBException 
{ 

try 
{ 

return rs.getlnt(columnName); 

} 

catch (SQLException oops) 
{ 

throw new DBException(oops); 

} 

} 

public int getlnt(int position) throws DBException 
{ 

try 

{ 

return rs.getlnt(position); 

} 

catch (SQLException oops) 

{ 

throw new DBException(oops); 

} 

} 

public InputStream getAsciiStream(String columnName) throws DBException 
{ 

try 
{ 

return rs.getAsciiStream(co!umnName); 

} 

catch (SQLException oops) 

{ 

throw new DBException(oops); 

} 

} 
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public short getShort(String columnName) throws DBException 
{ 

try 
{ 

return rs.getShort(columnName); 

} 

catch (SQLException oops) 

{ 

throw new DBException(oops); 

} 

} 

public boolean getBoolean(String columnName) throws DBException 

{ 

try 
{ 

return rs.getBoolean(columnName); 

} 

catch (SQLException oops) 
{ 

throw new DBException(oops); 

} } ^ ^ V 

public byte[] getBytes(String columnName) throws DBException 
{ 

try 
{ 

return rs.getBytes(coIumnName); 

> 

catch (SQLException oops) 
{ 

throw new DBException(oops); 

} 

public float getFloat(String columnName) throws DBException 

{ 

try 

{ . . 

return rs.getFloat(cohimnName); 

> 

catch (SQLException oops) 
{ 

throw new DBException(oops); 

} 

} 

public float getFloat(int position) throws DBException 
{ 

try 
{ 

return rs.getFloat(position); 

} 

catch (SQLException oops) 
{ 

throw new DBException(oops); 

} 

} 

public String getString(String columnName) throws DBException 
{ 

try 
{ 

return rs.getString(columnName); 
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} 

catch (SQLException oops) 
{ 

throw new DBException(oops); 

} 

} 

public Date getDate(String columnName) throws DB Exception 

{ 

try 
{ 

return rs.getDate(columnName); 

} 

catch (SQLException oops) 

{ 

throw new DBException(oops); 

public Timestamp getTimestamp(String columnName) throws DBException 
{ 

try 
{ 

• •- return rs.getTimestamp(columnName); - — ■■ « — — — - — ■•--> — ~ 

} • ...... 

catch (SQLException oops) - - . 

{ 

throw new DBException(oops); 

} 

} 

public boolean getBOFO throws DBException 
{ 

return atfiOF; 

> 

public boolean getEOFO throws DBException 
{ 

return atEOF; 
public void next() throws DBException 

{ • 

try 
{ 

atEOF = !rs.next(); 

> 

catch (SQLException oops) 

{ 

throw new DBException(oops); 

} 

} 

public boolean wasNullO throws DBException 
{ 

try 
{ 

return rs.wasNullO; 

} 

catch (SQLException oops) 

{ 

throw new DBException(oops); 

} 

} 

> 
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DJ 

package com. launch.PIaylistGenerator; 

public class DJ 

{ 

public int userlD; 

public String alias; 

public DJ (int id, String name) 

{ 

this(id); 
alias = name; 

} 

public DJ (int id) 
{ 

userlD = id; 

} 

} : 
DJ.java Page ! of 1 11/05/99 1:26 PM 
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DJList 

package com.launch.PlaylistGenerator; 

import java.utiLVector; 

public class DJList extends Vector 

{ 

public DJ djAt(int i) 
{ 

return (DJ) elementAt(i); 

} 

public String inList() 
{ 

Integer listfl - new Iriteger[size()]; 
int last = 0; 

for (int i = 0; i < this.sizeO; 
{ • 

- — ™~ — ~- - — — — list[i] = new Integer(dj At(i).userl D); ~ — ; - - — * - • 

>■ 

return Utiljoin(", list); 

} 

public boolean load(DBConnection conn, int userlD, int moodID) 
{ 

short djCount = 0; 

try 
{ 

DBResultSet rs = conn.executeSQL("exec sp_IcoGetDJs_xsxx " 

+ userID + M 
+ moodID); 

while (Irs.getBOFO && Irs.getEOFO) 
{ 

addElement(new DJ(rs.getInt("djID M ))); 

rs.nextO; 
djCount++; 

} 

Util.debug(Thread.currentThreadO-getNameO + " added " + djCount + " DJs M ); 

} 

catch (DBException oops) 
{ 

UtiI.debug("DB Exception in DJList::Ioad: " + oops.getMessageO); 

} 

return (djCount > 0); 

} 

public Vector asIDVectorO 
{ 

Vector users = new Vector(10); 



for (int i = 0; i < this.sizeO; 
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{ 

users.addElement(new Integer(((DJ) e!ementAt(i)).userID)); 

} 

return users; 

} 

} 

DJListjava Page 2 of 2 1 1/05/99 1:28 PM 
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FrequencyCounter 

package com.launch.PIaylistGenerator; 

import java.uti I.*; 

/** 

* FrequencyCounter is a Hashtable of the form (Object, Integer) 

* <br><br> 

* okay 1 realize the getLargest and getSmal lest Value 

* methods are very inefficient (CPU wise) but these methods 

* aren't called often, if they are then some one should 

* do an nlog(n) sort on them then just pick out the largest 

* after that 
**/ 

public class FrequencyCounter extends Hashtable 

{ 

public FrequencyCounter() 

{ 
} 

public FrequencyCounter(int i) 

super(i); . 



public void incrementValue(Object o) 
{ 

Integer i=(Integer)get(o); 
if(i=null) 

put(o, new Integer(l)); 

else 

put(o, new IntegerftLintValueO)*!)); 

} 



public FrequencyCounter getLargest(int n) 
{ 

FrequencyCounter fc=ne w FrequencyCounter(n+ 1 0); 

Integer temp_int; 

Object temp_object; 

Object smal iest_value Jeey=null; 

int smallest_value; 

Enumeration e=keysO; 

while (e.hasMoreElementsO) 
{ 

temp_object=e.nextEIeraentO; 
tempint=(Integer)get(ternp_object); 

if(fc.sizeO>=n) 
{ 

smallest_valueJcey^fc.getSmallestValueO; 

smailest_value^(lnteger)fc.get(smaIIest_value_key)).intValueO; 

if(temp_intintVaIue()>smallest_value) 

{ 

fc.remove(smallest_vaIue_key); 
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fc.put(temp_object, temp_int); 



} 

else 

{ 



fc.put(temp_object, tempint); 



} 

return(fc); 



/** @return null if list is empty */ 
public Object getSmallestValueO 
{ 

int smailest_vaIue=Integer.MAX_VALUE; 
Object smallest_value_key=null; 

int temp_int; 
Object temp_object; 



Enumeration e=keysQ; 

whi!e(e.hasMoreElementsO) 

{ 

temp_object=e.nextElementO; 
temp_int=((Integer)get(temp_object)).intValueO; 

if (temp_int<smallest_value) 
{ 

smallest_value=temp_int; 
smallest_value_key==temp_object; 

} 

} 

return(smailest_valuekey); 



yy******************************** ********* *************** 

II The following is a test function 

public static void main(String argv[j) 
{ 

FrequencyCounter fc==new FrequencyCounterO; 

fc.inc^ementValue("one ,, ); 
fc.incrementValueC'two"); 
fc.incrementValue( n two M ); 

fc.incrementValue("three M ); 
fc.incrementVaIue("three"); 
fc.uiaxmentValueC'three"); 

fc. increment VaIue("four"); 
fc.incrementVaIue("four n ); 
fc.incrementValue( M four"); 
fc.mcrernentValueTfour"); 
System.outprintln(fc); 

System.outprintln("smailest "+ fcgetSmallestValueQ); 
System.out.println( M largest T + fc.getLargest(2)); 

) 
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} 
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GeneratorParameters 

package com. launch. Play listGenerator; 

import javax.servlet.http.HttpServIetRequest; 

public class GeneratorParameters 

{ 

private int userlD, moodID, djID; 
private Bandwidth speed; 

private boolean debug, matrix, forceRefresh, dontsave; 
private MediaFormat format; 

private boolean moodlDSet = false; 
private boolean djIDSet = false; 



private int debugFormat - Util.DISPLAYTEXT; 
public Bandwidth speedO 
return speed; 



PCT/US00/30919 



public MediaFormat formatO 
return format; 

public int debugFormatO 

return debugFormat; 

public int userlDO 

return userlD; 

public int moodlDO 

return moodID; 

public int djIDO 

if(djlDSet) 

return djID; 

return userlD; 

public boolean debugO 
return debug; 

public boolean matrixO 
return matrix; 
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public boolean forceRefresh() 
{ 

return force Re fresh; 

} 

public boolean dontsaveO 
{ 

return dontsave; 

} 

public GeneratorParameters(HttpServletRequest request) 
{ 

debug = (request. getParameter("raIph") != null); 
matrix = (requestgetParameter^matrix") \= null); 
forceRefresh = (request.getParameter("forceRefresh") != null); 
dontsave - (request.getParameter("dontsave") != null); 

String debugFormatString = request.getParameterC'format"); 

if (debugFormatString != null && debugFo^natString.equals( M htrn^ , )) 
debugFormat = Util.DISPLAY_HTML; 



105 



try { userlD = Integer.parselnt(request.getParameter( H u ,l )); } 
catch (NumberFormatException e) { user ID = 0; } 

try { moodID = Integer.paneInt(request.getPararneter("rn")); } 

catch (NumberFormatException e) { moodID = 0; moodlDSet = false;} 

moodlDSet = true; 

try { djID = Intege^.pa^seInt(request.getParameter( ,, d ,, )); } 

catch (NumberFormatException e) { djID = userlD; djIDSet = false;} 

djIDSet = true; 

if(djID<=0) 
{ 

djID = userID; 
djIDSet = false; 

} 

speed = new Bandwid^request.getParameterC'b")); 
format = new MediaFormatO; 



> 
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Genrelndex 

package com. launch. PlaylistGenerator, 

import java.util.Hashtable; 

import java.util.Vector; 

public class Genrelndex extends Hashtable 

{ 

public Genrelndex(int x, int y) 
{ 

super(x,y); 

} 

public void add(short index, Songlnfo info) 
{ 

SongList list = get(index); 

if (list = null) 
{ 

list - new SongListO; 
put(new Short(index); list); 



list.addElement(info); 



} 



public SongList get(int index) 
{ 

return (SongList) get(new Short((short) index)); 

} 

public int countInGenreList(GenreList myGenres) 
{ 

int result = 0; 
SongList list; 

for (int i - 0; i < myGenres.sizeO; 
{ 

list = get(myGenres.genreAt(i)); 

if (list != null) 
{ 

result += list.sizeO; 

} 

} 

return result; 



* returns a COPY of the list of songs in genres 
V 

public SongList get!nGenreList(GenreList myGenres) 
{ 

SongList result = new SongListO; 
for (int i = 0; i < myGenres.sizeO; i++) 

{ 

resu!t.addElements(get(myGenres.genreAt(i))); 

} 
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Genrelndex 

package com.Iaunch.PlaylistGenerator; 

import java.util.Hashtable; 

import java.util. Vector; 

public class Genrelndex extends Hashtable 



public Genrelndex(int x, int y) 
{ 

super(x, y); 

} 

public void add(short index, Songlnfo info) 
{ 

SongList list = get(index); 

if(list = nul!) 
{ 

list = new SongListO; 
put(new Short(index), list); 



listaddElement(info); 



} 



public SongList get(int index) 
{ 

return (SongList) get(new Short((short) index)); 

} 

public int countInGenreList(GenreList myGenres) 
{ 

iiit result = 0; 
SongList list; 

for (int i = 0; i < myGenres.sizeO; 
{ 

list = get(myGenres.genreAt(i)); 

if (list != null) 
{ 

result += list.size(); 

} 

} 

return result; 



* returns a COPY of the list of songs in genres 
*/ 

public SongList getInGenreList(GenreList myGenres) 
{ 

SongList result = new SongListO; 

for (int i - 0; i < myGenres.sizeO; 
{ 

resuIt.addEIements(get(myGenres.genreAt(!))); 
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} 

/** 

* returns a COPY of the list of songs in a genre 
♦/ 

public SongList getInGenre(int genrelD) 
{ 

SongList list = get(genreID); 
SongList result; 

if (list = null) 

list = new SongList(); 
result = (SongList) listxloneO; 

return result; 
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GenreList 

package com. launch. PlaylistGenerator; 
import java.util.Hashtable; 
public class GenreList 

{ 

private int genres[]; 
private Hashtable hash; 

private byte next; 

public boolean allGenres = true; 

public GenreListO 
{ 

hash = new Hashtable(lJ); 
genres = new int[100J; 

} 

public int add(short genrelD) 

{ . ' 

— — -allGenres = false; - ~* — ~ -.~ — — - *™ — 

hash.put(new ShortCgenreiPX.new Boolean(true)); 

genres[next] - genrelD; ''" 

next++; 

return genresfnext - 1 ]; 

} 

public int sizeO 
{ 

return next; 

} 

public int genreAt(int pos) 

{ 

return genres[pos]; 

} 

public boolean exists(Short genrelD) 

{ 

if(next = 0) 

return true; 

else 

return hash.containsKey(genreID); 

} 

public String toStringO { 

String result = ,m ; 

for (int i = 0; i <size0; i++) 
{ 

result = result.concat(genreAt(i) + M , "); 

} 

return result; 

} 
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GetAds 

package com. launch. Play 1 istGenerator; 

import java.uti I. Date; 

import java.util. Vector; 

public class GetAds extends Thread 

{ 

Vector ads; 
int userlD; 
short mediaType; 



public GetAds(Vector ads, int userlD, short mediaType) 
{ 

this.ads = ads; 
this.userlD = userlD; 
this.mediaType - mediaType; 

} 

public void run() 
{ 

Date startDate = new DateO; 
ThreadxxuTentThreadO.setNarneC^GetAds"); 



int rowCount = 0; 

int count =0; " ■ " * • 

ClipaClip; 
int clipID, medialD; 
Date IastPlayed; 
String clipName; 

String sql = new String( M exec sp lcGetAds xsxx " 

+ userID 

+ -,- 

+ mediaType 

); 

try 

{ 

DBConnection conn = new DBConnectionO; 
DBResuItSet rs = conn.executeSQL(sql); 

while (Irs.getBOFO && Irs.getEOFO && count < Constants.MAX_ADS) 
{ 

ads.addElement(new Clip(rs.getInt( ,, clipID"), 

Clip.TYPE_AD, 
rs.getInt( M mediaID"), 
rs.getString("clipName"), 
rs.getDate("lastPlayed"))); 

count++; 
rs.nextO; 
rowCount-H-; 

} 

conn.closeO; 

} 

catch (DBException oops) 
{ 

UtiI.debug("DB Exception: " '+ oops.getMessageO); 

} 

Util.debug(ThreadxunrentThread0.getNameO+ " added M + count + " ads"); - 
UtiLprintElapsedTime(ThreadxurrentThread0.getNameO, startDate); 
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GetBDSStations 

package com.launch.PlaylistGenerator; 
import java.util. Date; 

public class GetBDSStations extends Thread 
{ 

int userlD; 
int moodID; 
Station Li st stations; 

public GetBDSStations(int userlD, int moodID, StationList stations) 

{ 

this.userlD = userlD; 
this.moodID = moodID; 
this.stations = stations; 

} 

public void run() 
{ 

Date startDate = new DateO; 

Threadxu^rentTh^ead0.setName( M GetBDSStations ,, ); 



int rowCount = 0; 

String sql = "sp_JcGetBDSNames_xsxx " +. userlD + w + moodID; 

try 

{ 

DBConnection conn = new DBConnectionQ; 

DBResu ItSet rs = conn.executeSQL(sql); 
while (!rs.getBOF0 && !rs.getEOF0) 
{ 

int bdsID = rs.getlntO'bdsID"); 
stations.addElement(new Station(bdsID)); 
rowCount++; 
rs.nextO; 

} 

conn.closeO; 

>■ 

catch (DBException oops) 
{ 

Util.debug("DB Exception in GetBDSStations: " + oops.getMessageO); 

} 

Util.debug(Thread.currentThread0^getNarneO + " got n + rowCount + " BDS station 

subscriptions"); 

Util.printEIapsedTime(Tru-eadxuiTentThreadO-getNameO» startDate); 

} 

} 

GetBDSStations.java Page 1 of 1 1 1/05/99 1 :38 PM 
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GetGenres 

package com.launch.PlaylistGenerator; 

import java.utiLDate; 

public class GetGenres extends Thread 

{ 

GenreList genres; 

intdjID; 

int moodlD; 

public GetGenres(GenreList genres, int djID, int moodlD) 
{ 

this.genres ~ genres; 
this.moodID = mood ID; 
this.dj[D==djlD; 

} 

public void runO 
{ 

Date startDate = new DateQ; 
^ _Threadxun-entThread0.setName< ,, GetGenres?); •■ w „. 



int rowCount = 0; - 



25 try 
{ 



DBConnection conn = new DBConnectionQ; 



DBResuItSet rs - connxxecuteSQLCexec sp_lcGetGenreNamesForUser_xsxx ' 
M +djID + ,, , ,t 

+ moodlD); 

while (Irs.getBOFQ && Irs.getEOFO) 
{ 

35 genres.add((short)rs.getInt( ,, genrelD")); 

rowCount++; 
rs.nextO; 

40 . conn.closeO; 
} 

catch (DBException oops) 

{ 

Util.debug("DB Exception: " + oops.getMessageO); 

45 } 

UtiI.debug(Thread.currentThread0.getNameO + " added n + rowCount + n genres"); 
Util.printElapsedTime(Ttoeadxum^ startDate); 

} 

50 } 

GetGenres.java Page 1 of 1 1 1/05/99 1 :38 PM 
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GetltemRatingsFromDB 

package com.Iaunch.PlaylistGenerator; 
import java.util.*; 

public final class GetltemRatingsFromDB extends Thread 

* { 

private Vector userlDs; 
private Vector results; 
//- 



public GetItemRatingsFromDB(Vector userlDs, Vector results) 
{ 

this. userlDs = userlDs; 
this, results = results; 

public void run0 
{ 

ThreadxurrentThread0.setName( w GetItemRatingsFromDB M ); 
Util.debug(ThreadxurrentThread0.getNameO + " thread started"); 
Date startDate = new DateQ; 



20 try 

~~ — ~ { ~-~ 



String sql = "SELECT iUserID_FK userlD, iSourceTablelD^L type, 
iItemID_FK itemID, tiRating rating FROM a 125ItemRating WHERE iUserID_FK IN (" + 
RatingsCache.GetVectorAsCommaDeIimitedList(userIDs) + *)'; 
25 DBConnection conn - new DBConnectionO; 

DBResuItSet rs = conn.executeSQL(sql); 
CachedRating cr; 

byte type; 

» while (Irs.getBOFO && Irs.getEOFO) 

{ 

cr « new CachedRating(rs.getInt( n userID"), rs^etlritfiternlD"), (byte) 
rs-getlntCrating"), sourceTablelDToTypeCrs.getlntC'type"))); 

results.addElement(cr); 
35 rs.nextO; 

} 

conn.closeO; 

} 

catch (DB Exception oops) 

40 { 

System.err.println("DBException in GetltemRatingsFromDB: " + 

oops.getMessageO); 

} 

Util.printElapsedTime(Thread.currentThreadOgetNameO, startDate); 

45 } 

public final static byte sourceTablelDToType (int type) 
{ 

if(type = 260) 

so return Constants JTEM_TYPE_ARTIST; 

// assume album (243) 
return Coristants.ITEM_TYPE_ALBUM; 

55 } 
} 

GetftemRatingsFromDB.java Page 2 of 2 1 1/05/99 1:32 PM 
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GetLastPIayed 

package com. launch. PlaylistGenerator; 

import Java. util. Date; 

import java.text. DateFormat; 

import javax.servlet.ServletOutputStream; 

public class GetLastPIayed extends Thread 

{ 

PlayDates lastPlayed; 
int userlD; 

ServletOutputStream out; 

public GetLastPlayed(PlayDates lastPlayed, int userlD, ServletOutputStream out) 
{ 

this. lastPlayed - lastPlayed; 
this.userlD = userlD; 
this.out = out; 

} 

public void runQ 

{ 

Date startbate = new DateO; 
Threadxur^entTh^ead0.setNahle( H GetLastPlayed ,, ); *- . ~- — 

// returns: songID, lastPlayed * 

try 

DBConnection conn = new DBConnectionO; 

UtiI.printElapsedTime(Thread.currentThread0.getNameO + " got a dbConnection", 

startDate); 

lastPlayed. load(conn, userlD); 

Util.printElapsedTime(Thread.currentThread0.getName() + " loaded dates", startDate); 
//this is somewhat expensive, so only do it every so often 

if(Util.random(10)=l) 
{ 

Util.debugfresaving lastPlayed for user " + userlD); 
lastPlayed.save(conn); 

} 

connxloseO; 

} 

catch (DBException oops) 
{ 

Util.debug("DB Exception: " + oops.getMessage()); 

} 

UtiI.out(out, ThreadxurrentThreadO-getNameO + M loaded " + lastPlayed.sizeO + " dates"); 
Util.printElaps^dTime(Thread.currentThreadO-getNameO + "done GetLastPIayed", startDate); 

} 

} 

GetLastPIayed.java Page 2 of 2 1 1/05/99 1 :35 PM 
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GetNews 

package com. launch. Play I istGenerator; 

import java.util.Date; 

import java.util. Vector; 

public class GetNews extends Thread 

{ 

Vector news; 
int userlD; 
short mediaType; 
int moodID; 

public GetNews( Vector news, int userlD, short mediaType, int mood ID) 
{ 

this.news - news; 
this.userlD = userlD; 
this.mediaType - mediaType; 
this, mood ID = moodID; 

} 

public void runO 
{ 

— ^ —Date startDate = new Date0; - * — ~~~™ — 

Th^eadxuJTentThread0.setName("GetNews ,, ); 



int rowGount = 0; 
int count = 0; 

ClipaClip; 
int clipID, medialD; 
Date lastPlayed; 
String clipName; 

/* 

spJcGetNews_xsxx @userID int, @moodID int, @mediaType int 
returns clipl D, clipName, medial D, lastPlayed 

*/ 

String sql = new Stringf'exec splcGetNewsxsxx n 

+ userlD 

_f. ■ 

+ moodID 

+ mediaType 

); 

try 
{ 

DBConnection conn = new DBConnectionO; 
DBResultSet rs = conn.executeSQL(sql); 

while(!rs.getBOF0 && Irs.getEOFO && count < Constants.MAX_NEWS JTEMS) 
{ 

news.addEIement(new GlipCrs.getlnt^cIiplD"), 

CJip.TYPE_NEWS, 
rs.getInt( ,, mediaID ,, ) ( 
rs.getStringC'clipName"), 
rs.getDate( w lastPlayed"))); 

count++; 
rs.nextO; 
rowCount++; 
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conn.closeO; 

} 

catch (DBException oops) 
{ 

Util.debug("DB Exception: " + oops.getMessageQ); 

} 

Util.debug(Thread.currentThreadO-getNameO + " added " + count + " news items"); 
Util.printEIapsedTime(Thread.currentThreadO getNameO, startDate); 

} 

} 

GetNews Java Page 2 of 2 1 1/05/99 1 :38 PM 
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GetPlaylist 

package com. launch. PlaylistGenerator; 

import java.util.Date; 

public class GetPlaylist extends Thread 

{ 

Population songs; 
int userlD; 

SonglnfoCache cache; 

public GetPlaylist(Population songs, int userlD, SonglnfoCache cache) 
{ 

this.songs = songs; 
this.userlD = userlD; 
this.cache - cache; 

public void runO 
{ 

Date startDate = new DateO; 
Th^ead.cu^^entTh^ead0.setNan^e("GetPlayIist ,, ); 

— _ r » — — Songlnfo info = null; — ~~ — - * — : — — — — — 

SirnpIeClip clip; 

intsongID; 

int rowCount = 0; 

try 
{ 

DBConnection conn = new DBConnectionO; 

Util.printElapsedTime(Thread.currentThread0.getNameO + " got a dbConnection", 

startDate); 

SimplePlaylist playlist = SimplePlaylist.Ioad(conn, userlD); 

if (playlist != null) 

{ 

for (int i ?= 0; i < play list, songs. sizeO; i++> 
{ 

clip - (SirnpIeClip) playlist.songs.elementAt(i); 
songID = clip.ID; 

songs.initSong(songID, Song.EXCLUDED); 

info = (Songlnfo) cache. get(songlD, SongInfoCache.TYPE_SONG); 

songs. artistCounts. increment info, album, artist. ID); 
songs.albumCounts.increment(info.album.lD); 

rowCount-H-; 

} 

} 

connxloseO; 

} 

catch (DBException oops) 
{ 

Util.debug( M DB Exception: " + oops.getMessageO); 

} 

UtiLdebug(Thread.currentThreadOgetNameO + " excluded " + rowCount + " songs"); 
Util.printElapsedTim^ThreadxurrentTru-eadO-getName startDate); 

} 

} 

GetPlaylist Java Page 2 of 2 1 1/05/99 1:34 PM 
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GetPIaylistServers 



package com. launch. Play I istGenerator; 
import java.util.*; 
/** 
**/ 

public final class GetPIaylistServers extends Thread 
{ 

public static int SLEEP_TIME = (3600*1000); // every hour 
public static int EXPECTED_SERVER_COUNT = 10; 
private GetPIaylistServersInterface personToNotify; 

/** 

* @param personToNotify must not be null. 
**/ 

public GetPlaylistServers(GetPlaylistServerslnterface personToNotify) 
{ 

this.personToNotiry=personToNotify; 

} 

public void run() 
{ 

— A . ™. Th^eadxurremTll^ead0.setName( w getPIaylistSe^vers ,, ); 



Util.debug(Thread.currentThreadO getName() + " thread started"); 
DBConnection conn; 
DBResultSet rs; 
25 Vector v; 

Date benchmark_date; 

try 

{ 

while (personToNotifyHnull) 

. { 

benchmark_date=new DateO; 

v-new Vector(EXPECTED_SERVER_COUNT); 

conn = new DBConnectionQ; 

rs = conn.executeSQL("exec spJcGetRatingsCacheServers_xsxd n ); 
35 while (Irs.getBOFO && !rs.getEOF()) 

{ 

v.addElement(rs.getString("server")); 
rs.nextO; 

} 

40 conn.closeO; 

personToNotify.updatePlaylistServers(v); 

Util.printElapsedTime(ThreadxuirentThread0.getNarneO + n , get w + 

v.sizeO + " rows", benchmark_date); 

Thread.sleep(SLEEP_TIME); 

} 

} 

catch (Exception e) 
{ 

System.err.println(new DateO-toStringO + " Fatal Exception in 

so GetPIaylistServers:" +e.toStringO); 

} 

Util.debug(Thread.currentThread0.getNameO + '* thread done"); 

} 

} 
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GetPlaylistServersInterface 

package com.Iaunch.PlaylistGenerator; 
import Java. uti I.*; 

public interface GetPlaylistServersInterface 
{ 

/** 

* @param playlistServers will be a vector of strings, each string is an ip address of the form 
xxx.xxx.xxx.xxx 

*♦/ 

public void updatePlaylistServers( Vector playlistServers); 

} 
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GetPopuIar 

package com. launch. Play IistGenerator; 

import java.util.Date; . ^ ? ♦* 

public class GetPopuIar extends Thread 

{ 

Population songs; 
SongList list; 

public GetPopuIar(Population songs, SongList list) 
{ 

this, songs = songs; 
this, list = list; 

} 

public void runO 
{ 

Date startDate = new Date(); 
Thread.cuJTentTh^ead0.setName( ,t GetPopula^ ,, ); 
Song ditty; 
SongData data; 
Songlnfo info; 



int rowCount = 0; 

if (list != null) 
{ 

for (int i = 0; i < list.sizeO; 
I 

info = listelerhentAt(i); 

data = songs.getSongData(info.songID); 

if(data!=null) 
{ 

// we can't add it, but let's append the info while we're here 
data.setInfo(info); 

} 

else 
{ 

data = songs.initSongGetData(info.songID, Song.UNRATED); 

if(data!=null) 
{ 

data.querySource = data.SOURCE_POPULAR; 
data.setInfo(info); * 

} 

rowCount++; 

} 

} 

} 

Util.debug(Thread.currentThread0.getNarneO + ° added " + rowCount + * songs"); 
Ud!.printElapsedTime(ThreadxuirentThread0.getNarneO, startDate); 

} 

} 

GetPopularjava Page 2 of 2 1 1/05/99 1:38 PM 
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GetRatings 

package com.launch.PlaylistGenerator; 

import java.util.Date; 

import java.utiL Vector; 

import java.util. Enumeration; 

import javax.serv let. ServletOutputStream; 

public class GetRatings extends Thread 

{ 

ItemsProfile profile; 
int user ID; 
DJList djs; 
Population songs; 
SonglnfoCache cache; 
ServletOutputStream out; 

public GetRatings(Population songs, ItemsProfile profile, int userlD, DJList djs, SonglnfoCache cache, 
ServletOutputStream out) 

: { 

this.prpfile = profile; 
this.userlD = userlD; 

this.cache = cache; 
this.songs = songs; 

public void run() 
{ 

Date startDate = new DateO; 
Thread.currentThreadO-setName( ,, GetRatings"); 

int rowCount = 0; 

// make a users vector from the users and djs 

Vector users = djs.aslDVector(); 
users.addElement(new Integer(userID)); 

Util.out(out, "GetRatings getting ratings for users " + users.toStringQ); 

Vector ratings = cache.ratingsCache.getRatings(users); 

UtihprintEIapsedTime("GetRatings after all ratings retreived", startDate); 

CachedRating cached; 
intdjID, itemID; 
byte rating, type; 
SongData data; 

short songType = Song.EXPLICIT; 
Songlnfo info; 
int artistID; 
Itemtheltem; 

int songRatings = 0; 
int itemRatings = 0; 

int userSongRatings = 0; 
int userltemRatings = 0; 
int djSongRatings = 0; 
int djltemRatings = 0; 

for (Enumeration e = ratings.elementsO; e.hasMoreElements() ;) 
{ 
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cached = (CachedRating) e.nextElementO; 

djID = cached. userlD; 

itemID = cached.itemID; 

rating = cached.rating; 

type = cached.type; 

// 0 is not a valid userld 

// ratings < 0 mean it was unrated 

if(djID != 0 1| rating < 0) 

{ 



if (type = 
{ 



= Constants. ITEM_TYPE_SONG) 

songRatings++; 

// store the user's rating 
if(userID = djID) 

< 

userSongRatings++; 



SongInfoCache.TYPE_SONG); 



SongInfoCache.TYPE_SONG); 
encoded 

Song.EXCLUDED); 



SongData.SOURCE_RATED; 
SongRating.RATING_SOURCE_EXPLICIT); 

this user for the artist 



if(rating = 0) 
{ 



} 

else 
{ 



} 



songs.iriitSong(itemID, Song.EXCLUDED); 
info = (Songlnfo) cache.get(itemID, 

addToAverageCinfo, 0); 



data = songs.initSongGetData(itemID, songType); 

if (data != null) 

{ 

info = (Songlnfo) cache.get(itemID, 



// if the song isn't in the cache, it's not 

// and we can*t play it 
if (info = null) 
{ 

songs. initSong( item ID, 

} 

else 
{ 

data.setln fo(info); 
data.query Source = 

data.rating.set(rating, 



// add this rating to all ratings by 
addToAverage(info, rating); 

} 

} 



} 

else // this is another user's song rating 
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dj$ongRatings++; 

data - songs.initSongGetData(item!D t Song. UNRATED); 



if(data!=null) 
{ 

data.querySource = SongData.SOURCE_DJS; 
data.djsAverage.add(rating); 



} 

} 

// don't count various artists ratings 
else if (!(type = Constants JTEMJIYPE_ARTIST && 
Artistlnfo.isVariousArtists(itemlD))) 

{ 



"^itemRatings-H-; ~ "** ! " 

theltem = profiIe.put(itemID); 

if(djID = userID) 
{ 

userltem Ratings++; 
theItem.userRating.set(rating); 



} 

else 
{ 



djItemRatings++; 
theItem.djsAverage.add(rating); 



rowCount-H-; 



} 



} 



Util.out(out, Hiread.currentThreadO getNameO + " added " 
+ songRatings + " song ratings (" 
+ userSongRatings + " user, " 
+ djSongRatings + " dj) " 
+ "and w + itemRatings + " item ratings (" 
+ userltemRatings + " user, " 
+ djltemRatings + " dj) w 

); 

Util.printElapsedTime(ThreadxurrentThread0.getNameO, startDate); 



private void addToAverage(SongInfo info, int rating) 
{ 

if(info!==nulI) 
{ 

(profi!e.put(info.aIbum.artist.ID)).songAverage.add(rating); 

} 

} 



private String userCriteriaO 
{ 
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its 

if (djs.sizeO <= 0) 

return " = " + userlD; 

return 'TN (" + userlD + M + djs.inListO + ")"; 

190 

} 

} 

GetRatings.java Page 4 of 4 1 1/05/99 1:35 PM 
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GetRatingsCacheUsers 

package com. launch. Play listGenerator; 
import java.util.*; 
importjava.net*; 
/*♦ 
**/ 

public final class GetRatingsCacheUsers extends Thread 
{ 

private static int SLEEP_TIME = (10 * 60 * 1000); // update every 10 minutes 
private static int EXPECTED_TOP_USER_SlZE =100; 
private GetRatingsCacheUsers Interface personToNotify; 

private static final int UPDATE J)B_C AC HED_USERS_SLEEP_COUNT = 6*8;// three times 
every day (6*8*SLEEP_TIME) 

// 

/** 

* @param personToNotify must not be null. 

public GetRatingsCacheUsers(GetRatingsCacheUsersInterface personToNotify) 
{ 

this.personToNotify = personToNotify; 
public void runO . 

{ - ' " ____ 

ThreadxurrentThreadOsetName("GetRatingsCacheUsers"); 

Util.debug(TlireadxurrentThreadO-getNameO + " thread started"); 

DBConnection conn; 

String mylP; 

DBResultSetrs; 

Vector v; 

Date bencfimark_date; 

try 

{ 

mylP = InetAddress.getLocalHostO-getHostAddressO; 
int update_db_users_list - 
UPDATE_DB_CACHED_USERS_SLEEP_COUNT; 

while (personToNotify != null) 

{ 

benchmark_date = new DateO; 

v = new Vector(EXPECTED_TOP__USER_STZE); 

conn = new DBConnectionO; 

rs = conn.executeSQL("exec sp_lcGetUsersToCache_isxd IM + mylP + 

•v 1 ); 

while (Irs.getBOFO && !rs.getEOF0) 
{ 

v.addE!ement(new Integer(rs.getInt( , 'use^ID ,, ))); 
rs.nextQ; 

} 

personToNotify.updateCachedUsers(v); 

Util.prmtElapsedTime(TlireadxurrentTriread0.getNameO + "» get " + 

v.size() + " rows", bench markdate); 

Thread.sleep(SLEEP_TIME); 
//— 

if (update_db__usersjist <- 0) 
{ 

// do the update 

Util.debug(new DateO-toStringO + " Updating 

RatingsCacheUserList"); 

try 
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{ 

Hashtable h = 

personToNotify.getMostFrequentIyUsedUsers(EXPECTED__TOP_USER_SIZE); 

if (h !=null && h.sizeO>0) 
{ 

String thecommand = "exec 

conn.executeSQL(the_command); 
Enumeration e = h.keys(); 
while (e.hasMoreElementsO) 
{ 

thecommand = "exec 
conn.executeSQL(the_command); 

} 



sp_IcDeleteRatingsCacheUsers_xxxd"; 



spJcAddRatingsCacheUserixxx " + e.nextElement(); 



} 

conn.closeO; 

} 

^^^r^-r^r^^- "^^"^catch (DBException dbe) ^ " " J ~- " • ~' r — ' — 

• • : . : r\..; A..' . - . ' " \ 

System.err.println(new DateO-toStringO + " 

DBException in GetRatingsCacheUsers: " + dbe.toStringO); 

dbe.printStackTraceO; 

} 

update_db_users_list = 
UPDATE_DB_CACHED_USERS„SLEEP_COUNT; 

} 

else 
{ 

Util.debug( n update_db_users_list is " + update_db_users_list); 
update_db_users_Iist— ; 

} 

//-- 

conn.closeO; 

} 

} 

catch (Exception e) 
{ 

System.err.println(new DateO-toStringO + " Fatal Exception in 
GetRatingsCacheUsers: " + e.getMessageO); 

e.priritStackTraceO; 

} 

Util.debug(Thread.currentThread0.getNameO + " thread done"); 

} 

} 
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GetRatingsCacheUsersInterface 

package com. launch. Play I istGenerator; 
import java.util.*; 

public interface GetRatingsCacheUsersInterface 
{ 

/** 

* @param topUsers will be a vector of Integers, where each integer is a userlD 
public void updateCachedUsers( Vector topUsers); 

/** 

* This method will return a hash of (Integer USERID, Intger Requests) 

* @param i is the number of users to get 

* @return null if no statistics 
**/ 

public Hashtable getMostFrequentlyUsedUsers(!nt i); 

} 
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GetRecentlyPlayed 

package com. launch. PlaylistGenerator; 
import java.util.Date; 

public class GetRecentlyPlayed extends Thread 
{ 

Population songs; 
int userlD; 

public GetRecentlyPIayed(Population songs, int userlD) 
{ 

this.songs = songs; 
this.userlD = userlD; 

} 

public void runQ 
{ 

Date startDate = new DateO; 

llireadxuiTentThxeadO^etNarneC'^etRecentlyPlayed"); 
int rowCount = 0; 

.. -~ ... ~ String sql^ new String("exec^ — - 

+ userID); 

int songID, albumID, artistlD; 
try 

DBConnection conn = new DBConnectionO; 
DBResultSet rs = conn.executeSQL(sql); 
while(!rs.getBOF0 && !rs.getEOF0) 
{ 

// returns songID, albumID, artistlD, lastPlayed 

albumID = rs.getlnt( M albumID"); 
songID - rs.getlntO'songlD"); 
artistlD = ^s.getInt( ,, a^t!St^D ,, ); 

// dont play these songs so soon again 
songs.initSong(songID,Song.EXCLUDED); 

songs.artistCounts.increment(artistID); 
songs.albumCounts.increment(albumID); 

rs.nextO; 
rowCount++; 

conn.closeO; 

} 

catch (DBException oops) 
{ 

Util.debug("DBException: M + oops.getMessageO); 

} 

Util.debug(Thread.currentThreadO getNameO + " added " + rowCount + M songs"); 
Util.printEIapsedTime(ThreadxurrentThread0.getNarneQ, startDate); 
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GetSonglnfoServIet 

package com. launch. Play listGenerator; 

import java.util.*; 

import java.io.*; 

importjava.net.*; 

import javax.servlet.*; 

import javax.servlet.http.*; 

/** 



* GetSonglnfoServIet 

* ©author Jeff Boulter 



*/ 

public class GetSonglnfoServIet extends HttpServlet 
{ 

public static final byte ONLINE_TIMEOUT = 10; 
/** 

* Handle requests... 
*/ 

public void doGet ( 

HttpServIetRequest request, 
HttpServletResponse response 
) throws ServIetException, IOException 



{ 



String user! D; 

String volume; 

String djlD; 

String djName; 

String djPosessive; 

String songName = ""; 

String albumName = ""; 

String artistName = " M ; 

int songID = 0; 

int aIbumlD = 0; 

intartistlD = 0; 

int comm Rating = 0; 

Date dateAdded = new DateO; 

byte origin - 0; 

int medialD = 0; 

int year = 0; 

intsongRating^-l; 

int albumRating=-l; 

int artistRating = -l; 

// get stream for output 

ServletOutputStream out; 

response.setContentType( M text/htmr); 

out = response.getOutputStreamO; 

responscsetHeaderCTragma' 1 , "no-cache"); 

response.setHeader("Cache-control "no-cache"); 

responscsetHeaderC'Expires", "0"); 

try 



{ 



userlD = requestgetParametere'rater"); 

if(userID = nu!l) 

{ 

out.print!n("no userlD passed"); 
return; 

} 
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DBConnection conn = new DBConnectionO; 
djlD = request.getParameter("djID"); 
djName - request.getParameter("djName"); 
if (djlD = null || djlD.equals(userlD)) 
{ 

djName = "You"; 
djPosessive = "Your"; 

} 

else 
{ 

djPosessive = djName + "V; 

} 

DBResultSet rs = conn.executeSQLC'exec spJcGetPlayinglnfoForUserxsxx " 

while (!rs.getBOF() && !rs.getEOF()) 
{ 

songName = rs.getString("song tt ); 
albumName = rs.getString("album H ); 
artistName = rs.getString( M artist"); 
songID - rs.getlnt("songID"); 
albumID = rs.getlntC'albumlD"); 
artistlD « rs.getlnt("artistID n ); 
commRating = rs.getInt( H commRating"); 
if (commRating <= 0) { commRating = - 1 ;} 
origin = (byte) rs-getlntCorigin"); 
medialD = rs.getlntO'medialD"); 
year = rs.getlnt^year"); 
dateAdded = rs.getTimestamp("dateAdded"); 
songRating = rs.getInt("songRating"); 
albumRating = rs.getInt( n aIbumRating"); 
artistRating = rs.getint( n artistRating"); 
rs.nextO; 

. } 

int exclusive = isExclusive(albumName); 
int. newStatus = isNew(dateAdded); 
int popular = isPopular(commRating); 
String djs = ""; 

if (origin = SongData.SOURCE_DJS__ALBUM) 
djs = djRatings(conn, userlD, albumID, 

Constants. ITEM_TYPE_ALBUM); 

else if (origin = SongData.SOURCE_DJS_ARTIST) 
djs = djRatingsfconn, userlD, artistlD, 

Constants.ITEM_TYPE_ARTIST); 

else 

djs - djRatings(conn, userlD, songID, 

Constants.ITEMTYPESONG); 



out.print( 



us formatAIbumYear(year)) + 



djPosessive)) + "& w 



"media_id- H + medialD + 
+ n song L id=" + songID .+ 
+ "songjiarrie- ' + escape(songName) + "&" 
+ "album_id=" + albumID + M &" 
+ "album_name=" + escape(albumName + 

+ "artist Jd=" + artistlD + 

+ "artistname^" + escape(artistName) + 

+ "exclusive^" + exclusive + 

+ "comm_rating=" + commRating + "&" 

+ "new=" + newStatus + 

+ "origin- 1 + escape(SongData.originText(origin, djName, 
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+ "popu!ar=" + popular + "& M 

+ "song_ratingr" + songRating + "&" 

+ "songratingtype^P + M &" 

+ "a]bum_rating= ,f + albumRating + 

+ "album jatin&_type=r + 

+ "artist_rating=" + artistRating + 

+ t, artist_rating_type=l" 

+ djs 

+ fans(conn, songID) 

+ radioStations(conn, userlD, songID) 

+ n &ticker_text=&image_url=" // not used 

); 

volume = requestgetParameter("volume M ); 
saveVolume(conn, userlD, volume); 
conn.closeO; 

catch (DBException e) 

System.err.printlnC'DBException: " + e.getMessageO); 
e.printStackTraceO; 

catch (Exception e) 

out.printIn("Exception raised: " + e); 
e.printStackTraceO; 

outxloseO; 

} 

private void saveVo!ume(DBConnection conn, String userlD, String voIumeStr) throws 

DBException 

{ 

if (voIumeStr — null) 

return; 
double volume - 0; 
try 
{ 

Double dblVolume = new Double(volumeStr); 
if (dblVolume !=» null) 

volume -db I Volume.doubleValueO; 

catch (Exception e) . . 

return; 

if (volume > 0 && volume <= 100) 

conn.executeSQLC'exec spJcSetVolume_isux " + userlD + " + volume); 



private String djRatings(DBConnection conn, String userlD, int itemID, String storedProc, String 
variableName) throws DBException 
{ 

String result = ""; 
String djName; 
String ratingStr; 
int rating; 
int count - 1 ; 

DBResultSet rs « conn.executeSQLC'exec " + storedProc + n n + userlD + H , M + itemID); 

while (frs.getBOFO && !rs.getEOF()) 

{ 
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185 rating = rs.getlnt("rating"); 

if (rating <= 0) 
{ 

ratingStr = "X"; 

} 

190 else 

{ 

ratingStr = "" + rating; 

} 

result = result.concat( 

las + variableName + "_name" + count + "=" + 

escape(rs.getString("alias")) 

+ "&*' + variableName + "_id" + count + + rs.getlnt("userID") 
+ "& M + variableName + "_value" + count + + ratingStr 
+ + variableName + '^online" + count + "=" + 

200 isOnline(rs.getInt("minutesSincePlay")) 

); 

count++; 
rs.next0; 

j 



205 



210 DB Exception 



return result; 

} 

private String djRatings(DBConnection conn, String userlD, int itemID, byte itemType) throws 
{ 



if (itemType = Constants.ITEM_TYPE_SONG) 
{ 

return djRatings(conn, userlD, itemID, 
2i5 "sp_lcGetUserDJRatingsForSongID_xsxx M , n dj_rating H ); 

} 

else if (itemType = Constants. ITEM_TYPE_ALBUM) 
{ 

return djRatings(conn, userlD, itemID, 
220 "sp_lcGetUserDJRatmgsForAlburnID_xsxx M , "dj_rating M ); 

> 

else if (itemType = Constants.ITEM_TYPE_ARTIST) 
{ 

return djRatings(conn, userlD, itemID, 
as "sp_IcGetUserDJRatingsForArtistID_xsxx", "dj_rating"); 

} 

return ""; 

} 

230 

private String radioStations(DBConnection conn, String userlD, int songID) throws DBException 

{ 

int count = 0; 

String result = ""; ■ 
235 DBResuItSet rs - conn.executeSQLf'exec 

spJcGetSubscribedBDSStationsPlayingSongxsxx M + userlD + M + songID); 

while (!rs.getBOF0 && !rs.getEOF()) 
{ 

result = result. concat( 

240 ,, &radio_id ,, + count + + re.getInt("bdsStationID") 

+ ,t &radio_name" + count + ' -" + escape(rs.getString("callLetters M ) + " 

" + rs.getString("description H )) 

); 

count++; 

245 rs.nextO; 

} 

App. 2-67 



WO 01/35667 PCTAJSOO/30919 

92 

return result; 

} 

private String fans(DBConnection conn, int songID) throws DBException 
{ 

String result = 
int count = 1 ; 
int rating; 

String ratingStr = ,m ; 

DBResultSet rs = conn.executeSQL( M exec sp_lcGetFans_xsxx M + songID); 

while (frs-getBOFO Irs.getEOFO && count <= 5) 

{ 

result = resuh.concat( 

"&fan_name M + count + ' + escapeOrs.getStringCalias")) 
+ "&fan_id" + count + ' + rs.get!nt("userID") 
+ M &fan_on!ine" + count + "=" + 

isOnIine(rs.getInt( M minutesSincePlay M )) 

); 

count++; 
rs.nextO; 

} 

if (count > 1 && !rs.getEOF0) 
{ 

result = result.concat( f, &fianjd M + count + M =0 rt + "&fan_nanie" + count + 



n =more... w ); 



} 

return result; 

> 

private String formatAlbumYear(int year) 
{ 

if(year>0) 
{ 

return " (" + year + ")"; 

} 

return ""; 

} 

private int isExclusive(String albumName) 
{ 

if (albumName != null) 
{ 

if (albumName.indexOf("Launch Live") > -1) 
{ 

return I; 

} 

} 

return 0; 

} 

private int isOnline (int lastPlay) 
{ 

if (ONL1NE TIMEOUT > lastPlay) 
{ 

return 1; 

} 

return 0; 

} 

private int isPopular (int commRating) 
{ 

if (commRating > Constants.POPULAR_THRESHOLD) 
{ 

return I; 

return 0; 
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private int isNew (Date dateAdded) 

{ 

if(dateAdded = null) 
{ 

return 0; 

} 

long two Weeks = UtiI.MILLISECONDS_IN_SECOND * 

Util.SECONDS_IN_MINUTE * 
UtiI.MINUTES_IN_HOUR ♦ 
UtiI.HOURS_lN_DAY * 
14; 

Date now = new DateO; 

if (now.getTimeO - date Added. getTimeO < two Weeks) 
{ 

return 1; 

return 0; 

} 

private String escape(String thing) 
{ 

if(thing = null) 
return""; 

} 

return URLEncoder.encode(thing); 

} 

public void init (ServIetConfig config) 
throws ServletException 

{ 

super, in it(config); 

} 

public void destroy() 

{ 
} 

/*eof*/ 

GetSonglnfoServletjava Page 8 of 8 11/05/99 1:38 PM 
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GetSongRatingsFromDB 

package com.launch.PlaylistGenerator; 
import java.util.*; 

public final class GetSongRatingsFromDB extends Thread 
{ 

private Vector userlDs; 
private Vector results; 



public GetSongRatingsFromDB(Vector userlDs, Vector results) 
{ 

this. userlDs = userlDs; 
this.results = results; 

} 

public void run() 
{ 

ThreadxurrentThread0. setName("GetSongRatingsFrom DB"); 
Util.debug(Thread.currentThreadOgetNarneO + "thread started"); 
Date startDate = new DateQ; 

try 

String sql = "SELECT iUserf D_FK userlD, iSongID_FK songID, iRating rating 
FROM a200SongRating WHERE iUserID_FK IN (" + RatingsCache.GetVectorAsComrnaDelimitedList(userIDs) + 

J; 

DBConnection conn = new DBConnectionO; 
DBResultSet rs = cohn.executeSQL(sql); 
CachedRating cr; 

while (rrs.getBOF0 && !rs.getEOF()) 
{ 

cr = new CachedRating(re.getInt( M userID"), rs.getlntC'songlD"), 
(byte)rs.getlnt("rating")i Constants.ITEM_TYPE_SONG); 

results. addElement(cr); 
rs.next(); 

} 

conn.closeO; 

} 

catch (DBException oops) 
{ 

System.err.println(" DBException in GetSongRatingsFromDB: M + 

oops.getMessageO); 

} 

Util .printElapsedTime(Thread.currentThread0.getNameO, startDate); 

} 

} 

GetSongRatingsFromDB.java Page I of 1 1 1/05/99 1 :32 PM 
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IntHash 

package com.launch.PiaylistGenerator; 

import java.uti 1 . HashtabJe; 

/** 

* A hashtable that uses ints as keys and values. 
*/ 

public class IntHash extends Hashtable 
{ 

public synchronized int get(int key) 
{ 

Object thing = get(new Integer(key)); 

if (thing = null) 
return 0; 

else 

return ((Integer) thing).intVaiue(); 

} 

public synchronized int put(int key, int value) 

{ 

put(new IntegenTcey), new Integer(value)); 
return value; 

} ' 

private synchronized int change(int key, int valueChange) 
{ 

return put(key, get(key) + valueChange); 

} , 

public synchronized int increment int key) 
{ 

return change(key, 1); 

} 

public synchronized int decrement(int key) 
{ 

return change(key, -1); 

} 

public synchronized int increment(int key, int howMuch) 
{ 

return change(key, howMuch); 

} 

public synchronized int decrement(int key, int howMuch) 
{ 

return change(key, -howMuch); 

} 



IntHashjava Page 1 of 1 1 1/05/99 1 :26 PM 
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Item 

package com. launch. Play 1 istGenerator; 

public class Item 

{ 

public final static byte TYPE_ANY = 0; 
public final static byte TYPE_ALBUM = 1 ; 
public final static byte TYPEARTIST = 2; 
public final static byte TYPE UNKNOWN - 10; 

public int item ID; 

public Rating userRating; 

private boolean songAvgScoreCalcuIated = false; 

private double songAvgScore; 

// the average rating from all djs for this tiem 
public AverageRating djsAverage; 

// average rating of all songs by an artist 

- public AverageRating songAverage; ' - - - ^ _ . - - 

public double songAverageScore(ArtistInfo info) 
{ 

if (! songAvgScoreCalcuIated) 
{ 

songAvgScoreCalcuIated = true; 

double songsByArtist = Math.min(info.songs.sizeO, 
Constants.MAX_SONGS_BY_ARTIST); 

double songsRated ~ Math.min(songAveragexount(), 
Constants.MAX_SONGS_BY_ARTIST); 

// deviation from the average 

songAvgScore = ((songAverage.getO - Constants.DEFAULT_RATING) 

* (songsRated / songsByArtist)) + Constants.DE F AULT_RATING ; 

} 

return songAvgScore; 

} 

public boolean inGenres - false; 

public byte getTypeO 
{ 

if(itemID = 0) 

return TYPE_UNKNOWN; 
else if (itemID < 1000000) 

return TYPE_ALBUM; 

else 

return TYPE_ARTIST; 

} 

public String typeNameO 
{ 

byte type = getTypeO; 

if (type = TYPE_ALBUM) 
return "Album"; 

App. 2-72 



WO 01/35667 

97 

else if (type = TYPE_ARTIST) 
return "Artist"; 

else 

return "Unknown"; 

} 

public ItemO 
{ 

userRating = newRatingO; 
djsAverage = new AverageRatingO; 
songAverage = new AverageRatingO; 

} 

public Item(int itemID) 
{ 

this(); 

this jtemlD = itemID; 

} 

public String toString(Song!nfoCache cache) 

String title = "(Not available)"; 
byte type = getTypeQ; 



PCT/US00/30919 



if (type = TYPE_ARTIST) 
{ 

Artistlnfo artist = (Artistlnfo) cache.get(item[D, SongInfoCache.TYPE_ARTIST); 

if (artist != null) 

title = artist.title; 

> ■ 

else if (type = TYPE_ALBUM) 
{ 

Albumlnfo album = (Albumlnfo) cache.get(itemID, SonglnfoCache.TYPE ALBUM); 

if (album !- null) 

title ~ album.title; 

} 

return typeNameO + " V ,B + title + <V (" + itemID + n ) " 
+ "user=" .+ userRating.toStringO 
+ " djs=" + dj sA verage.toStringO 
+ rt songAverage=" + songA verage.toStringO 
+ " songAvgScore=" + songAvgScore; 



} 

Itemjava Page 2 of 2 1 1/05/99 1 :24 PM 
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ItemsProfile 

package com. launch. PlaylistGenerator; 

import Java. util.Hashtable; 

import java.utiL Enumeration; 

import javax.servlet.ServletOutputStream; 

public class ItemsProfile 

{ 

private Hashtable hash; 

public ItemsProfiIe() 
{ 

hash = new HashtableO; 

} 

public synchronized Item get(int itemID) 
return get(new Integer(itemID)); 

} 



public synchronized Item get(Integer itemID) 
{ 

return (Item) hash.get(itemlD); 

} 

/** 

* puts a new item in the hash and returns it. 

* If it's already there, just return it 
*/ 

public synchronized Item put(int itemID) 
{ 

Integer ID = new Integer(itemID); 

Itemit = get(ID); 

if (it = null) 
{ 

it = new Item(itemlD); 
hash.put(ID, it); 
return it; 

) 

else 

return it; 

} 

public void print(ServletOutputStream out, SonglnfoCache cache) 
{ 

for (Enumeration e = hash.keysO; e-hasMoreElementsQ ;) { 
Item anltem = get((Integer) cnextEIementO); 
UtiI.out(out, anItem.toString(cache)); 

} 

} 

public String inList(byte type) 
{ 

String list = " M ; 

for (Enumeration e = hash.keysO; e hasMoreEIementsQ ;) { 
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Item anltem = get((Integer) e.nextEIementO); 

if (type = Item.TYPE_ANY || anltem .getTypeO = type) 
{ 

list = list.concat(an Item, item ID + V); 

} 

} 

// remove that extra comma 
if(list.length()>0) 

list = list.substringCOj list.IengthO - 1); 

return list; 

} 

} 

ItemsProfile Java Page 2 of 2 1 1/05/99 1 :32 PM 
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Media 

package com. launch. Play listGenerator, 
public class Media 

{ 

int medial D; 
short mediaType; 
String filepath; 

public Media(int medialD, short mediaType, String filepath) 
{ 

this.medialD = medialD; 
this.mediaType = mediaType; 
this.filepath = filepath; 

} 

public String toStringO 
{ 

return mediaType + M + medialD; 

} 

— ~* — public static short getMediaType(Bandwidth speed, MediaFormat format) 
{ 

if (formatgetO = MediaFormat WINDO WSMEDI A) 
{ 

if(speed.get() = Bandwidth.SPEED_28) 
return 211; 

else if (speed.get() — Bandwidth.SPEED_56) 
return 147; 

else if (speed.getO >= Bandwidth.SPEED_l00) 
return 212; 

else 

return 0; 

} 

return 0; 

} 

public static Bandwidth typeTbBandwidth(short mediaType) 
{ 

if(mediaType = 2Il) 

return new Bandwidth(Bandwidth.SPEEDJ28); 
else if (mediaType = 147) 

return new Bandwidth(Bandwidth.SPEED_56); 
else if (mediaType = 212) 

return new Bandwidth(Bandwidth.SPEED_l 00); 

return new BandwidthO; 

} 

} 

Media.java Page 1 of 1 1 1/05/99 1 :28 PM 
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MediaFormat 

package com. launch. PI ay IistGenerator; 

public class MediaFormat 

{ 

public final static byte WINDOWSMEDIA = I ; 
public final static byte REALMEDIA « 2; 
public final static byte QUICKTIME = 3; 

private boolean beenset = false; 

private byte value; 

// when we start supporting more than one format, just take this out 

public MediaFormatO 

{ 

value - WINDOWSMEDIA; 
beenset = true; 

} 

— „ public MediaFormat(byte format) — — — ~ — 

{ 

value = format; 
beenset = true; 

} 

public byte getO 
{ 

return value; 

} 

public void set(byte format) 
{ • 

value = format; 
beenset = true; 

} 

public boolean isSetO 
{ 

return beenset; v 

} 

publ ic String toStringO 
{ 

if (value = WINDOWSMEDIA) 

return "WindowsMedia"; 
else if (value = REALMEDIA) 

return "RealMedia"; 
else if (value = QUICKTIME) 

return "QuickTime"; 

return "UNKNOWN"; 

} 

} 

MediaFormatjava . Page 1 of 1 1 1/05/99 1 :25 PM 
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MediaGatewayServlet 

package com. launch. PlaylistGenerator; 

import java.io.*; 

importjava.net.*; 

import javax.servlet.*; 

import javax.servlet.http.*; 

import java.util.*; 

/•♦ 

* , 

* PlaylistGeneratorServIetJava 8/16/99 

* Servlet that redirects to media 

* Copyright (c) 1999 Launch, Inc. 

* @author Jeff Boulter 

V 

public final class MediaGatewayServlet extends HttpServIet 
{ 

/** what browser signature we look for */ 

private static final String mpSignature = "NSPlayer"; 

--■/** when we get an unauthorized browser, play this */ — — ,.,„„, , , . — 

private static final String unauthorizedBrowser = "audio/errors/unauthorizedbrowser.asf *; 
/** when we get an unauthorized user, play this */ 

private static final String unauthorizedUser = "audio/errors/unauthorizeduser.asf '; 

/** when we get an unauthorized user, play this */ 

private static final String outOfMedia = "audio/errors/outonnedia.asf '; 

/** how many tries we take to get media */ 

private static final int MAXJTERATIONS = 5; 

/** this is the header that media player uses toe indicate which query it is */ 
private static final String CONTEXTTAG = "request-contexts"; 

/** To work around a problem with reading multiple headers with the same name in servlet 2.0 + jrun, 
look for these headers to determine the context */ 

private static final String FIRST_REQUEST_PRAGMA = "xClientGUID"; 
private static final String SECOND_REQUEST_PRAGMA = "stream-switch-entry**; 
private static final String REQUEST_CONTEXT = "request-context="; 
private static final int STREAMING_MEDIA_TIMEOUT=1000*60*15; 
/♦♦ 

* Handle requests... 
*/ 

public final void doGet (HttpServletRequest request, HttpServletResponse response) throws 
ServletException, lOException 
{ 

// Util.debug("MediaRedirectServIet:doGetO received a request"); 

DBConnection conn = null; 
ServletOutputStream out = null; 
int context; 
intuserID = -l; 
boolean debug=false; 
try 
{ 

// get connections and streams 

conn = new DBConnectionO; 

out = response.getOutputStreamO; 

// get parameters from http 

debug = (requestgetParameter^'ralph") != null); 

// setup response data 

setResponseHeaders(response); 

setResponseContentType(response, debug); 

// get parameters from http 

userlD = Integer.parseInt(request.getParameter( ,, u")); 

if (!checkUserAgent(request.getHeader("USER_AGENT"), debug, out)) 
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{ 

return; 

} 

// muck with clip and clip schedule 

ClipSchedule schedule = new ClipScheduIe(userlD); 

schedule, in it(conn); //db call 1 

Clip aClip^null; 

int iteration; 

boolean done = false; 

// keep going until we get a good path 

for (iteration - 0; iteration < MAXITERATIONS && !done; iteration-H-) 
{ 

aClip = new Clip(schedule.nextCIipType(debug, out)); 

if (aClip — null || aClip.typeO = Clip.TYPE_NONE) 
{ 

done = true; 

System.err.println("user M + userlD + " is out of songs to play"); 



} 

„~ else 
{ 



// get the paths and stuff 
aClip.getPath(conn, schedule); // db call 2 
if(aClip.is$etO) 
< 

done = true; 

} 

else 
{ 

done = true; 

System.err.println("user " + userlD + " is out of media of type 



" + aCIip.typeNameO + "to play"); 
* } 

} 

} 

// update the playlist 
ioo schedu!e.pIaylist.save(conn, userlD); // db call 3 

if(aClip = null) 

out.println(Constants.STREAM_SERVER + 7" + outOfMedia); 

else 

105 { 

// log the play 

aClip.logPlay(conn, userlD); // db call 4 

no //get the URL 

outprintln(aCHp.URLO); 

} 

} 

catch (NumberFormatException e) 
{ 

out.println( H Bad userld"); 

// print out the MMS path to redirect to 

if (debug) 

{ 

i2o out.println( H redirecting to n + unauthorizedliser); 

} 

else 
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} 



out.println(Constants.STREAM_SERVER + *T + unauthorizedUser); 



e.getMessageQ); 



catch (Throwable e) 
{ 

System. err.printIn("Generic Exception in MediaGateway for userlD " + user ID + ": " + 



} 

finally 
{ 



first 32 bytes.) 



e.printStackTrace(); 

try 
{ 

if(out!=null) 
{ 

out.cIoseO; 

} 

if (conn!=nulI) 
{ 

conn.closeO; 

catch (SocketException se) 

// donl do anything, the person disconnected, no error, (or mediaplayer sampled 

catch (Exception el) 

e 1 .printStackTraceQ; 



> 



} 



private final boolean checkUserAgent(String agent, boolean debug, ServletOutputStream out) throws 
lOException 
{ 

if (!(agent!=null && agent.startsWith(mpSignature))) 
{ 

if (debug) 
{ 

out.println("in valid useragent. Would stream " + unauthorizedBrowser); 
return true; 

} 

else 
{ 

out.println(Constants.STREAM_SERVER + 7" + unauthorizedBrowser); 

} 

return(false); 

} 

else 
{ 



} 



retum(true); 



} 

private final void setResponseContentType(HttpServletResponse response, boolean debug) 
{ 

if (debug) 
{ 

response.setContentType("text/plain M ); 

} 

else 
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{ 

response.setContentType("video/x-ms-asr); 

} 

} 

private final void setRe$ponseHeaders(HttpServletResponse response) 
{ 

response.setHeader("Pragma", M no-cache"); 
response.setHeaderC'Cache-control", M no-cache M ); 
response.setHeader( M Expires", M 0"); 

} 



private static final void readFi!eToOutputStream(String filename, HttpServletResponse response, boolean 
{ 



/* 

debug) 

{ 

readFileToOutputStream(new File(filename), response, debug); 

} 

private static final void readFileToOutputStream(File thefile, HttpServletResponse response, boolean 

debug) 

{ 

try 
{ 

„„BufferedInputStream bis=new BufferedInputStream(new FileInputStream(the -file));- 



BufferedOutputStream bos=new BufferedOutputStreairi(response.getOutputStrearn()); 
bosiflushO; //this is to ward off any problems 1 think there might be a jrun problem with : 
initializing the output stream fast enough, i.e. before we get there... 

BufFeredWriter br=new Buffered Writer(new OutputStream Writer(bos)); 
if (debug) 

Util.out(response.getOutputStream0, "streaming file H + the_file + " of size M + 

the_file.lengthO); 

else. 

response.setContentLength((int)the_file.lengthO); 
// System.err.println("streaming file " + the_file + n of size " + the_file.IengthO); 
RedirectStream redirecting_stream=new RedirectStream(bis, bos, debug, 
response.getOutputStreamO); 

redirecting_stream.startO; 

redirecting^stream.join(STREAMmG_MEDIA_TIMEOUT, 0); 
if (redirecting_stream.isAHveO) redirecting^stream.stopO; 
//System.eiT.printin^finished streaming"); ^ 

} 

catch (SocketException se) 
{ 

// don't do anything, the person disconnected, no error, (or mediaplayer sampled first 32 

bytes.) 

} 

catch (FileNotFoundException fe) 
{ 

System.erT.println( M readFileToOutputStream could not find file " + the_file + " for 
reading:" + fe.getMessageO); 
} 

catch (Exception e) 
{ 

e.printStackTraceO; 

} 

} 

private int getContext(HttpServletRequest request) 
{ 

try 
{ 

String pragma = requestgetHeader^'pragma"); 
// Util.debug("pragma is " + pragma); 

if (pragma = null) 
return 0; 
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int index = pragma. indexOf(REQUEST CONTEXT); 
// Util.debugfindex is " + index); 

if (index <0) 
{ 

return 0; 

} 

else 
{ 

int start = index + REQU ESTCONTEXT. Iength() ; 
String contextNum = pragma.substring(start, start + 1); 
// UtiLdebug( n contextNum is " + contextNum); 

return Integer.parseInt(contextNum); 

} 

// when I can read multiple headers with the same name I should use the below code 
// int Iocation=pragma.indexOf{CONTEXT_TAG); 

// Iocation=Iocation+CONTEXT_TAG . Iength(); 

// int lastlocation; 

// for (lastjocation=location; last_location<pragma.length() && 

pragma.charAt(last_location)!-/; lasr location++); 
// retum(lntegerj>aralnttoragm 

> 

w ~ * — —catch (Exception e) , , , , — ^ , « — 



{ 

Util.debug( M Exception caught in getContext: " + e.toStringO); 
return 0; 

} 

} 

*/ 
} 

MediaGatewaySeFvIetjava Page 7 of 7 1 1/05/99 1:24 PM 
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MediaList 

package com.Iaunch.PlaylistGenerator; 
import java.util. Vector; 
public class MediaList 
{ 

private Vector media = new Vector(0, 1); 

public void add(short mediaType, int medialD, String filepath) 
{ 

media.addElement(new Media(mediaID, mediaType, filepath)); 

public boolean inType(short mediaType) 
{ 

Media test; 

for (int i = 0; i < media.sizeO; 'i-h-) 

{ ; 

test = (Media) media.eIementAt(i); 

if (test.mediaType = mediaType) 
return true; 

} 

return false; 

} 

public int getID(short mediaType) 
{ 

for (int i = 0; i < media.sizeO; i++) 
{ 

Media aMedia = (Media) media.elementAt(i); 

if (aMedi&mediaType — mediaType) 
return aMedia.medialD; 

} 



} 



return 0; 



public String getFilepath(short mediaType) 
{ 

for (int i = 0; i < media.sizeO; i++) 

Media aMedia = (Media) media.elementAt(i); 

if (aMedia.mediaType = mediaType) 
return aMedia.fi lepath; 

} 



} 



return null; 



public int sizefj 
{ 

return media.sizeO; 

} 
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public Media typeAt(int index) 
{ 

return (Media) media.eiementAt(index); 

} 

public String toStringO 
{ 

String result = 

if (media = null) 

return "(none)**; 

for (int i = 0; i < media.sizeO; i+ + ) 
{ 

result - resuIt.concat(media.elementAt(i).toStringO + ","); 

} 

return + result + ")"; 

} 

} 

MediaListjava Page 2 of 2 11/05/99 1:28 PM 
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PickCount 

package com. launch. Play I istGenerator; 
import javax.servlet.ServletOutputStream; 
/** 
*/ 

public class PickCount 
{ 

int explicit; 
int implicit; 
int unrated; 
String method = ""; 

public PickCount(int userlD, int djID, int ratio, int playlistSize, Population songs, ServletOutputStream 

out) 

{ 

float explicitSize - songs.explicit.sizeO; 
float implicitSize = songs.implicit.size(); 
float unratedSize = songs.unrated.sizeO; 

UtiLout(out, "Available: explicit songs: " + explicitSize + implicit songs: " + implicitSize + ", 
unrated songs: " + unratedSize); 

; ^ . JUtil.out(out, "Ratio: ZJl ratio); , cii ^ ,; u , ^ m 

// if you're listening to someone else's station, try to not listen to any unrated songs 

if (userlD — djID) 

{ 

// let's try to use their ratio 

double totalRated « (explicitSize + implicitSize); 

if (totalRated < Constants;MIN_RATINGS_TO_HONOR_RATIO) 

method = "New User Unrated Ratio"; 

ratio = Constants.NEW_USER_UNRATED_RATIO; 

int maxPIicit = (int) Math.round(playlistSize * ( 1 00 - ratio) * 0.0 1 ); 
int maxRatedToPick = (int) Math.round(expIicitSize * 
Constants.MAX_PERCENT_RATED_SONGS_TO_PICK * 0.01); 

// pick three times as much from rated 

int explicitToPick = (int) Math.round(playIistSize * (100 - ratio) * 0.01 * (explicitSize / 

totalRated) * 3); 

int impIicitToPick = maxPIicit - explicitToPick; 

explicit = (int) Math.min(maxRatedToPick, explicitToPick); 

implicit = (int) Math.min(implicitSize, impIicitToPick); 

// pick up the slack in unrated 

unrated = (playlistSize - explicit - implicit); 

method = "Unrated Ratio"; 

} 

// if you're listening to someone else's station and they have enough ratings, 
// don't play unrated 

else if ((explicitSize + implicitSize) > Constants.MIN_SlZE_FOR__NO_UNRATED) 

explicit = (int) Math.round(playlistSize * 0.50); 
explicit = (int) Math.round(Math.min(explicit, (explicitSize * 
Constants.MAX_PERCENT_RAXED_SONGS_TO_PICK) * 0.01)); 

implicit = (int) Math.min(pIaylistSize, implicitSize) - explicit; 

method = "DJ play - no unrated"; 

// if we didn't get enough, use the default method 

if (explicit + implicit < playlistSize) 

{ 

explicit « (int) Math.round(pIaylistSize * 0.33); 
explicit = (int) Math.round(Math.m inexplicit, (explicitSize * 
Constants.MAX_PERCENT_RATED_SONGS_TO_PICK) / 1 00.0)); 

implicit = (int) Math.round(pIayIistSize * 0.33); 

implicit = (int) Math.round(Maui.min(implicit, (implicitSize * 
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Constants.MAX_PERCENT_RATED_SONGS_TO_PICK) / 1 00.0)); 

unrated = playlistSize - explicit - implicit; 
method = "DJ play - not enough rated"; 

} 

} 

// if neither of these worked 

else 

{ 

explicit = (int) Math.round(playlistSize * 0.33); 

explicit = (int) Math.round(Math.min(explicit, (explicitSize * 
Constants.MAX_PERCENT_RATED_SONGS_TO_PICK) / 100.0)); 

implicit = (int) Math.round(playlistSize * 0.33); 

implicit = (int) Math.round(Math.min(implicit, (implicitSize * 
Constants.MAX_PERCENT_RATED_SONGS_TO_PlCK) / 100.0)); 

unrated = playlistSize - explicit - implicit; 

method = "Default 33/33/33 method"; 

} 

Util.out(out, "Picking: explicit songs: " 

+ explicit j 
+ implicit songs: " 
+ implicit 

~~ — ~ — ~ — unrated songs: - — - — — ~ 

+ unrated 

+ method = " + method 

); 

} 

public String toStringO 
{ 

return "explicit to pick: " 
+ explicit 

+ implicit to pick: " 
+ implicit 

+ unrated to pick: " 
+ unrated; 

} 

public void resetO 
{ 

explicit = 0; 
implicit = 0; 
unrated ~0; 

} 

PickCountjava Page 3 of 3 1 1/05/99 1 :24 PM 
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PickList 

package com. launch. Play I istGenerator; 

import javaaitil.Vector; 

public class PickList extends Vector 

{ 

public PickList(PickCount counts) 
{ 

// make a list of all the song types that we need to pick 
for (int i = 0; i < counts.explicit; i++) 

addElement(Song.EXPLICIT); 
for (int i = 0; i < counts. implicit; 

addElement(Song.IMPLICIT); 
for (int i = 0; i < counts. unrated; i++) 

addElement(Song.UNRATED); 

} 

public void addElement(short value) 

{ 

addEIement(new Short( value)); 

} \ .' " 

-~ —-public void reAdd (short type, Vector songGroup, Population songs)- 

{ _ 
// try to pick from the same bucket again 

if (songGroup.sizeO > 0) 

addElement(type); 
// otherwise, try the other ones 
else if (songs.explicit.sizeO > 0) 

addElement(Song.EXPLICIT); 
else if (songs.implicitsizeO > 0) 

addElement(Song.lM PLICIT); 
else if (songs.unrated.sizeO > 0) 

addElement(Song.UNRATED); 

• } 

public short getRandomO 
{ 

if(size0<0) 

return 0; 

int lucky = (int) Util.random(sizeQ - 1 ); 

// figure out what group to pick from 

short type - ((Short) element At(lucky)).shortValue(); 

removeElementAt(lucky); 

return type; 

) 

} 

PickListjava Page 2 of 2 1 1/05/99 1 :27 PM 
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PickStatus 

package com.launch.PlaylistGenerator; 

public class PickStatus 

{ 

5 public final static int NOTJMCKED = 0; 

public final static int REJECTED = 2; 
public final static int PICKED = 1 ; 

int status; 
10 int order = -1; 

short percentile; 

public String toString() 
{ 

is return toDisplayString(Util.DISPLAY_TEXT); 

} 

public String toDispIayString(int dispIayType) 

{■'.'■ 

.20 _ ; ... ^String.redStart ,^.. m .„,,„ „.„ ,,.„, , _ 

String greenStart = 1 
String fontEnd = ' 



t== m.. 



if (dispIayType — Util.DISPLAY^HTML) 
{ 

redStart = "<FONT COLOR=red><B> n ; 
greenStart = M <FONT COLOR=greenxB>"; 
fontEnd = " < VBx/FONT> H ; 

} 

switch (status) { 

case NOT_PICKED: 

return M N "; 
case PICKED: 

return greenStart +" P " + fontEnd; 
case REJECTED: 

return redStart + " R" + fontEnd; 

default: 

return * 

} 

} 

} 

PickStatus.java Page 1 of I 1 1/05/99 1 :26 PM 
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PlavDataHash 

package com. launch. P lay listGenerator; 

import java.util. Enumeration; 

public class PlayDataHash extends IntHash 

{ 

public String toStringO 
{ 

String myString = ""; 



for (Enumeration e - keysO; e.hasMoreElementsO ;) { 

// debug.wri teO'interation " + i++); 

int stationID = ((Integer) e.nextElement()).intValueO; 

int rank = get(stationlD); 

myString = myString.concat( 

"stationID: " + 
stationID + 

rank + 

... _ ___:^n n ); 



return myString; 

} 

} 

PlayDataHash.java Page 1 of 1 1 1/05/99 1 :26 PM 
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PlayDates 

package com. launch. Play I istGenerator; 
import java.util.Hashtable; 
import java.util. Date; 
import java.util.Enumeration; 
import java.text.SimpleDateFormat; 
import java.io.InputStreamReader; 
import java.text. Parse Pos it ion ; 
import java.io.IOException; 
import java.utif. Calendar; 
public class PlayDates 



private static final String dateFormat = "yyyy-MM-dd HH:mm:ss"; 

private Hashtable hash; 
int userlD; 

double secondsInDay = Util.MILLISECONDSJN_SECOND * 

- - ^ — Util.SECONDS^rN^MINUTE*- 

Util.MINUTES_IN_HOUR * 
Util.HOURS_IN_DAY; 

// for date parsing 

private static StringBuffer year = new StringBuffer( n 1 234"); 
private static StringBuffer month = new StringBuffer(" 12"); 
private static StringBuffer day = new StringBuffer( M 1 2 M ); 

private static StringBuffer hour = new StringBufferCn"); 
private static StringBuffer minutes - new StringBufferC* 1 2 n ); 
public Date dbDate = new E>ate{); 

private boolean loaded = false; 

public PlayDatesO 
{ 

hash = new HashtableO; 

} 

public void put(int songID, Date lastPIayed) 
{ 

// the common case is that they will have NOT played this song before, 
// so create the Integer object in anticipation that we will use it for 
// the put as well. 

Integer i = new Integer(songID); 

Date before = get(i); 

// save only the most recent play of a song 

if (before = null || before.getTimeO < IastPIayed.getTime()) 
{ 

hash.put(i, lastPIayed); 

} 

} 

public Date get(int songID) 
{ 

return (Date) hash.get(new Integer(songID)); 
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} 

public Date get(Integer songID) 
{ 

return (Date) hash.get(songlD); 

} 

public Enumeration keys() 
{ 

return hash.keys(); 

public void remove(Integer songID) 
{ 

hash.remove(songlD); 

} 

public int sizeO 
{ 

return hash.sizeO; 

} 



public String toString() 

{ . _ . - - ■ 

String result = ""; 

for (Enumeration e = hash.keysO; e.hasMoreElementsQ ;) { 
Integer songID = (Integer) e.nextEIementO; 
Date playedAt = get(songID); 

result = result.concat("{" + songID + w + playedAt + "} w ); 

} 

return result; 

} 

public String toDBStringO 
{ 

Date startDate = new DateO; 

StringBuffer buffer = new StringBuffer( 1 00000); 

Calendar cal = Calendar.getlnstanceO; 

Integer songID; 
Date playedAt; 

for (Enumeration e = hash.keysO; e.hasMoreElementsO ;) { 
songID = (Integer) e.nextEIementO; 
playedAt = get(songID); 

// System.out.println(playedAt); 

cal.setTime(playedAt); 

buffer.append(caLget(Calendar.YEAR) + 

+ leadingZero(caI.get(CaIendar.MONTH) + I) + 
+ leadingZero(cal.get(Calendar.DAY__OF_MONTH)) 
+ leadingZero(cal.get(Calendar.HOUR_OF_DAY)) + 
+ leadingZero(cal.get(CaIendar.MINLrTE)) + ":00= H 4 

songID + V); 
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// result = resultxoncat(formatter.format(playedAt) + ' + songID + ","); 

} 

Util.printElapsedTimeC'toDBString", startDate); 
return bufFer.toStringO; 

} 

public static final String leadingZero (int value) 
{ 

if (value < 10) 

return "0" + value; 

return value + 

} 

public float getScore(Integer songID) 
{ 

Date lastPlayed = get(songID); 
if (lastPlayed = null) 

' 1LJ . ~ — ~ return Or ~ . — ™. ... ~ — - 

double secondsSincePlayed = new Date().getTimeO - lastPlayed.getTimeO; 
double daysSincePlayed = secondsSincePlayed / secondsInDay; 
double logValue = Math.log(daysSincePlayed + 0.0 1); . 
return (float) Math.min(100, (22.0 * logValue)); 

} 

public void save(DBConnection conn) 
{ 

// Date dateStarted = new DateO; 

if(!loaded) 

return; 

try 
{ 

conn.executeUpdate("exec sp_icSavePlayHistoryTexf_isux " + iiserlD + '" + 

toDBStringO + n "\ false); 

} 

catch (DBException e) 

{ 

System.err.printIn("DBException in PlayDates:save: " + e.toStringO); 

} 

// UtiI.printEIapsedTime("save", dateStarted); 

} 

public void markRecentlyPIayed(SongInfoCache cache, Population songs) 
{ 

double now = dbDate.getTimeO; 

double lastThreeHours = Util.MILLISECONDS_IN_SECOND* 

Util.SECONDS_IN_MINUTE * 
Util.MrMJTESJNJiOUR * 
3; 

Integer songID; 
Date playedAt; 
Songlnfo info; 
int artist! D, albumID; 

for (Enumeration e = hash.keysO; e.hasMoreEIementsO ;) 
{ 
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songID - (Integer) e.nextElementO; 
playedAt = get(songID); 

if (now - playedAtgetTimeO < lastThreeHours) 
{ 

// mark songs played in the last three hours 

// so as to comply with the RJ AA rules 

// and make sure we don't pick too many later 

info = (Songlnfo) cache. get(songI D, SonglnfoCache.TYPE_SONG); 

if (info != null) 
{ 

artistID = info.getArtistID(); 
aibumID = info.getAlbumID(); 

// "various artists" albums don't count 

if(!ArtistInfo.isVariousArtists(ajtistID)) 

{ 

songs.artistCounts.increment(artistID); 

} 

songs.album Counts, incremental bum ID); 



} 

public void o!dLoad(DBConnection conn, int userlD) 
{ 

this.userlD = userlD; 

try 
{ 

String sql = "exec spJcoGetLastPIayed^xsxx " + userlD; 
DBResultSet rs = corm.executeSQL(sql); 

loaded = true; 

Date lastDate; 
int songID; 

while (Irs.getBOFQ && !rs.getEOF0) 
{ 

songID = rs.getlntO'songlD"); 

lastDate = rs.getTimestamp("lastPlayed"); 

put(songID, lastDate); 

rs.nextO; 

} 

} 

catch (DBException e) 
{ 

System.err.println("DBException in PlayDates.oidLoad: " + e.toStringO); 

} 



} 

public void load(DBConnection conn, int userlD) 
{ 
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Date startDate = new DateO; 

// be carefiil of the SQL Server TEXTS1ZE parameter which is by default 64KB 
this.userlD^userlD; 

double aDay = Util.MILLISECONDSJN_SECOND * 

Util.SECONDSJN_MINUTE * 
UtiI.MINUTES_lN_HOUR * 
Util.HOURS_IN_DAY; 

double aMonth = aDay * U ti 1 . D A YS_IN_MONTH ; 



try 
{ 



String sql = "exec splcGetSongHistoryTextxsxx " + userlD; 

DBResuItSet rs = conn.executeSQL(sql); 
UtiI.printElapsedTime< ,, LP: getsonghistorytext", startDate); 



. if (Irs.getBOFO && !rs.getEOF0) . , 

{ 

loaded = true; 

char[] stuff = new char[100000]; 

InputStreamReader reader = new 
InputStreamReadeiXrs.getAsciiStream^played")); 

Util.printElapsedTime("LP: created reader" , startDate); 
dbDate = rs.getTimestampC'dbDate"); 
long dbDateTime = dbDate.getTimeO; 

reader.read(stuff); 

Uti!.printEIapsedTime( w LP: read into stuff 1 , startDate); 
Calendar cal = Calendar.getlnstanceO; 
int lastStart = 0; 
intsongID = 0; 

// Si mpleDateFormat formatter 1 = new 

SimpleDateFormat(PlayDates.dateForrnat); 

// ParsePosition pos = new ParsePosition(0); 

Date datePIayed = null; 
String parseme = new StringO; 

long length = stuff.Iength; 

for (int i = 0; i < length; i++) 
{ 

switch (stuff[i]) 
{ 

case -': 

// parseme = new String(stuff, lastStart, i - lastStart); 

// pos.setlndex(O); 

// datePIayed = formatter l.parse(parseme, pos); 

datePIayed = parseDate(stuff, lastStart, cal); 
System.out.println( M date is " + datePIayed); 
if(datePlayed==nuII) 

<■ 

pos.setlndex(0); 
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datePlayed = formatted. parse(parseme, pos); 



lastStart = i + 1 ; 
break; 



case 



datePlayed.getTimeO) < aMonth)) 



// 

startDate); 



case 0: 



parseme = new String(stuff, lastStart, i - lastStart); 



songID = Integer.parselnt(parseme); 



try 
{ 

} 

catch (NumberFormatException e) { } 
// save 'em 

// also don't save them if they're > 30 days old 

if (songID > 0 && datePlayed != null && ((dbDateTime 



put(songID, datePlayed); 



{ 
} 

songID = 0; // reset' 

datePlayed = null; // reset 

lastStart 88 i 4-' l; 
break; 



// we're at the end of the string 
Util.printElapsedTimeC'LP: found null at char n + i, 

return; 



} 

} 

catch (DBException oops) 

{ 

Util.debug("DBException in PlayDates.load: " + oops.getMessage()); 

} 

catch (lOException oops) 
{ 

Util.debug("IOException in PlayDates.load: " + oops.getMessageO); 

} 

} 

/** 

* Why? Because SimpleDateFormat is *way* too slow. 
*/ 

private final Date parseDate(char[] chars, int start, Calendar cal) 
{ 

// 1999-10-13 17:19:00 
// 0123456789012345678 
f* 

String year, month, day, hour, minutes; 
year - new String(chars, start, 4); 
month = new Strihg(chars, start + 5, 2); 
day = new String(chars, start + 8, 2); 

hour = new String(chars, start +11,2); 
minutes = new String(chars, start +14, 2); 
*/ 
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year.setCharAt(0, chars[start + 0]); 
year.setCharAt(I, charsfstart + 1]); 
year.setCharAt(2, charsfstart + 2]); 
year.setCharAt(3, charsfstart + 3]); 

month.setCharAt(0, charsfstart + 5]); 
month.setCharAt(l, charsfstart + 6]); 

day.setCharAt(0, charsfstart + 8)); 
day.setCharAttI, charsfstart + 9)); 

hour.setCharAt(0, charsfstart +11]); 
hour.setCharAt(t, charsfstart + 12]); 

minutes.setCharAt(0, charsfstart + 14]); 
minutes.setCharAt( 1 , charsfstart + 1 5]); 

int yearlnt = 0, monthlnt = 0, daylnt = 0, hourlnt = 0, minuteslnt = 0; 
// try 
// { 

yearlnt = parselnt(year); 

^ „ ^ , _ mont h] nt = parseInt(month); - - - : ^ — — ; - - ~ - - ~- 

daylnt - parselnt(day); 

hourlnt = parse!nt(hour); 
minuteslnt = parselnt(minutes); 

// } 

// catch (NumberFormatException e) { return.null;} 

// cal.clearO; 

cal.set(yearlnt, monthlnt - 1, daylnt, hourlnt, minuteslnt, 0); 
return cal.getTimeQ; 



private static final int parseInt(StringBufFer s) 
{ 

int result - 0; 

int last = s.lengthO - 1 ; 

for (int i = last; i >= 0; i») 
{ 

result += char2int(s.charAt(i)) * Math.pow(10, last - i); 

return result; 

private final static int char2int(char ch) 
{ 

switch (ch) 
{ 



case T: 






return 1; 


case *2': 






return 2; 


case '3': 






return 3; 


case '4': 






return 4; 


case *5': 






return 5; 
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case '6': 






return 6; 


case T: 






return 7; 


case 






return 8; 


case '9': 






return 9; 


default: 






return 0; 



} 

} 

} 

PlayDatesjava Page 9 of 9 1 1/05/99 1 :35 PM 
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Playlist 

package com. launch. PlaylistGenerator; 
import java.uti I. Vector; 
import java.utiLHashtable; 
import java.utiLEnumeration; 
import java.utiLDate; 
public class Playlist 

{ 

Vector media; 

Vector news; 

Vector ads; 

Vector tips; 

int ID; 

int userlD; 

int djID; 

int moodID; 

short mediaType; 

boolean debug; 

boolean popuIarOnly = false; 

— * PickCount counts; — ~ ^ — ; — " ~ r ' ■^ ^■-i-T^-x-r^ -> . r "■' — ™— 

public final static int BUCKE^^ 

private int lastlndex; 

intbuckets[J; 

IntHash artists; 

IntHash albums; 

public PlaylistO 

{ 

artists = new IntHashQ; 

albums = new IntHashO; 

counts - null; 

media =newVector(); 

news == new Vector(); 

ads =newVector(); 

tips =newVectorO; 

buckets = new int[BUCKET_COUNT]; 

lastlndex = -1; 

debug = false; 

} 

public Playlist(int playlistID) 
{ 

thisO; 

ID = playlistID; 

} 

public void resetSourcesO 
{ 

for (int i = 0; i < BUCKET_COUNT; i++) 
bucketsfi] = 0; 

} 

private void saveOrigtns(DBConnection conn) 
{ 

String listString =""; 

SongData data; 

for (int i = 0; i < media.size(); 

{ 

listString = listString.concat(((SongData) media.elementAt(i)).originTcIListO); 

} 

try 
{ 

conn.executeSQLOexec spJcSaveOriginsJxxd " + userlD + m + listString + 
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catch (DBException oops) 
{ 

Util.debug("DB Exception: " + oops.getMessageO); 

} 

} 

public Playlist2 toPlaylist2() 
{ 

Playlist2 result = new Playlist2(); 
// copy playlist 

for (int i = 0; i < media.sizeO; i++) 

{ 

resultsongs.addElement(((SongData) media.elementAt(i)).toPIayIistEntry(mediaType)); 

} 

// copy news 

for (int i = 0; i < news.sizeO; i++) 
{ 

result.news.addEiement(((CIip) news.elementAt(i)).toPlaylistEntry(mediaType)); 

} 

// copy ads 

for (int i = 0; i < ads.sizeO; 
{ 

,... r — * * . .. resu!t^tds.addElement(((Clip) ads.e!ementAt(i)).toPIaylistEntry(mediaType));"-^^ — ■ 

;. : ..}..- . . 

//copy tips 

for (int i = 0; i < tips.sizeO; 
{ 

resu!t.tips.addElement(((Clip) tips.eIementAt(i)).toPlaylistEntry(mediaType)); 

} 

return result; 

} 

public String toStringO 
{ 

IntHash artistCount = new IntHashO; 

IntHash albumCount = new IntHashO; 

IntHash querySource = new IntHashO; 

Hashtable querySourceName - new HashtableO; 

IntHash originSource = new IntHashO; 

Hashtable originSourceName = new HashtableO; 

Hashtable artistNames - new HashtableO; 

Hashtable albumNames = new HashtableO; . • 

String result = "Playlist " + ID + " for userlD " + userlD 

+ " (djID " + djID + ") in mood " + moodlD 

+ " with mediaType " + mediaType 

+ ", pickCounts: " + counts 

+ " has " + media.sizeO + " songs:" 

+ Util.newLine; 

for (int i = 0; i < media.size0; 
{ 

SongData data = (SongData) media.elementAt(i); 
String songStr = data.getMedialD(mediaType) + " " 

+ data.getAIbumIDO + " " 

+ data.getArtistIDO + " " 

+ data.songID + " " 

+ data.getArtistNameO + " " 

+ data.getAlbumNameO + " " 

+ data.getSongNameO + Util.newLine; 
querySource.increment(data.querySource); 
querySourceName.put(new Integer(data.queryS6urce), 
a.sourceString(data.querySource)); 

byte origin = data.originO; 
originSource.increment(origin); 

originSourceName.put(new Integer(origin) f data.sourceString(origin)); 
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artistCount.incremcnt(data.getArtistlDO); 
albumCount.increment(data.getAlbumlDO); 
if (data.getArtistName() != null) 

artistNames.put(new Integer(data.getArtistIDO), data.getArtistNameO); 
if (data.getAIbumNameO != null) 

albumNames.put(new Integer(data.getAlbumlDO), data.getAIbumNameO); 
result = result.concat(songStr); 

} 

result = result.concat(Util.newLine); 

for (Enumeration e <= artistCount.keys(); e.hasMoreElementsO ;) { 
int artistID = ((integer) e.nextElementO)intValueO; 
String artistStr = artistCount.get(artistlD) 

+ n songs are by the artist " 
+ artistNames.get(new Integer(artistlD)) 
+ M (" + artistID + ") n 
+ Util.newLine; 

result = resultxoncat(artistStr); 

} 

result = result.concat(Util.newLine); 

for (Enumeration e = alburnCountkeysQ; e.hasMoreElementsO 0 { 
int albumlD = ((Integer) e.nextElementO).intValueO; 

— • — — ™™ — String albumStr = a!bumCount.get(albumID) - : — 

+ " songs are from the album " 
+ albumNames.get(new lnteger(albumlb)) 
+ "(" + albumlD + ")" 
+ Util.newLine; 

result = resu!tconcat(albumStr); 

} 

result = resu!t.concat(UtiLnewLine); 

for (Enumeration e = querySourcckeysQ; e.hasMoreElementsO 0 { 
int source = ((Integer) e.nextEIementO).intValueO; 
int songCount = querySource.get(source); 
double doubleCount = new Double(songCount).doubleVaiueO; 
String str = songCount 

+ " songs (" 

+ ((doubleCount / lengthO) * 100) 

+ "%) are from the" 

+ querySourceName.get(new 

Integer(source)) 

+ " query" 
+ Util.newLine; 

result = result.concat(str); 

} 

result = result.concat(UtiI.new Line); 

for (Enumeration e = originSource.keysO; e.hasMoreElementsO ;) { 
int source = ((Integer) e.nextElementO)-intVaIue(); 
int songCount = originSource.get(source); 
double doubleCount = new Double(songCount).doubleValueO; 
String str = songCount 

+ " songs (" 

+ ((doubleCount / lengthO) * 100) 

+ n %) originated from M 

+ originSourceName.get(new 

Integer(source)) 

+ Util.newLine; 

result = resultconcat(str); 

} 

result = resu!txoncat(UtiI.newLine); 
int bucketSize = 100 / BUCKETCOUNT; 
double playlistLength = media.sizeO; 
for (int i = 0; i < BUCKET_COUNT; i++) 
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result = result.concat( 

"Percentile " 

+ (i ♦ bucketSize) +"%-" 

+ ((i + 1) * bucketSize) + "%: " + buckets[i] + H (" 

+ Utilfix(100 * (buckets[i] / playlistLength), 2, 0) + "%)" + 

Util.newLine); 

} 

return (result + Util.newLine); 

} 

public int length 0 
{ 

return media.size(); 

> 

public void append (SongData song) 
{ 

float bucketSize = (new Float(l 01)).floatValue() / (new Float(BUCKET_COUNT)).floatValue(); 
int bucket = (int) Math.floor(song.status.percentile / bucketSize); 
// Util.debug("adding medial D n + song.medialD 

II + " in percentile " + song.status.percentile + M (bucket " 

// + bucket + T); 

media.addElement(song); 

buckets[bucket]++; — — ~ '<*-~*~*~~^ ~ l~ ~*m — 

. } ■ . - : , - . , . - , • y .y 

public Playlist shuffleO 
{ 

Vector newList = new Vector(media.sizeO); 

int rand = 0; 

while (mediELsizeO > 0) 

{ ' 
rand = (int) Util.random(media.size() - 1); 
Object m = media.elementAt(rand); 
mediaxemoveElementAt(rand); 
newListaddElement(m); 

} 

media = newList; 
return this; 

} 

public int nextOrdinaI(DBConnection conn) 
{ 

int ordinal = 1; 

try * 
{ 

DBResultSet rs = conn.executeSQL("exec spJcGetOrdinalID_xsxx " + userlD); 

while (irs.getBOFO && frs.getEOFO) 

{ 

ordinal = rs.getInt( w ordinar); 
rs.nextO; 

} 

conn.executeSQL("exec sp_lcUpdatePlay!istData_ixxd " 

+ userID + M , H 
+ djlD + V 
+ moodID + V 
+ mediaType); 

} 

catch (DBException oops) 
{ 

Util.debug("DB Exception in Playlist::nextOrdinal: " + oops.getMessage()); 

} 

return ordinal; 

public void deleteHighOrdinals(DBConnection conn, int ordinal) 
{ 
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try 
{ 

conn.executeSQLf'exec spJcDeletePlaylistRangexxxd *' 

+ userlD + " 
+ ordinal); 

} 

catch (DBException oops) 
{ 

UtiI.debug("DB Exception in PIayIist::de!eteHighOrdinals: " + oops.getMessageO); 

} 

} 

private SimplePlaylist toSimplePlaylistO 
{ 

SimplePlaylist result = new SimpIePlaylistO; 
result.mediaType = this.mediaType; 
result.djID = this.djlD; 
resultmoodID = this.moodID; 

// copy playlist 

for (int i = 0; i < media.sizeO; i++) 
{ 

- -result. songs.addElement(((SongData)media.elementA^ 



} 



} 

//copy news 

for (int i = 0; i < news.sizeQ; 
{ 

resu!t.news.addEIement(((Clip) news.elementAt(i)).toSimpleClip(mediaType)); 

} 

// copy ads 

for (int i - 0; i < ads.sizeO; 
{ 

result.ads.addElement(((Clip) ads.element At(i)).toSimpleCI ip(mediaType)); 

} 

// copy tips 

for (int i = 0; i < tips.sizeO; i++) 
{ 

result.tips.addElement(((CIip)tipsxIementAt(i)).toSimpleClip(rnediaType)); 

} 

return result; 



public void save (DBConnection conn, SimplePlaylist oldPIaylist) 

Date startDate = new DateQ; 

SimplePlaylist thoreau = toSimplePlaylistO; 

Util.printElapsedTime( M Convert to SimplePlaylist", startDate); 

if (oldPIaylist != null) 
{ 

thoreau.lastAd = oldPlaylist.lastAd; 
thoreau.lastNews = oldPlaylist.lastNews; 
thoreau. las tTip = oldPIaylist. lastTip; 

} 

thoreau.save(conn, user ID); 
UtiLprintElapsedTimeC'SavePIaylist", startDate); 

} 
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userlD); 
userlD); 
userlD); 



public boolean save (DBConnection conn) 
{ 

if(length() <= 0) 

return false; 
boolean resetOrdinal = false; 
int highOrdinal, ordinal; 
Date startDate = new DateO; 
highOrdinal = ordinal = nextOrdinal(conn); 
if (highOrdinal > MAX_ORDINAL) 
{ 

ordinal = 1; 
resetOrdinal - true; 

} 

UtiLpruitElapsedTimeCGetOrdinar', startDate); 

Thread saveNews = new SaveClips(news, "spJcSaveNewsPlaylisMxxd", ordinal, mediaType, 

Thread saveAds = new SaveClips(ads, n sp_lcSaveAdsPlayIist_ixxd", ordinal, mediaType, 

Thread saveTips = new SaveClips(tips, "spJcSaveTipsPIaylistjxxd", ordinal, mediaType, 

int partition = (int) Math.round(mediasize() / 4.0); 

* ' Thread savePlaylistl = new SavePlaylist(this;0,* partition, ordinal); ~ ,- ™; — ~~" *" 

Thread savePlaylist2 = new SavePlayIist(th]s, partition, partition * 2, ordinal + partition); 
Thread savePIaylist3 "- new SavePIaylist(this, partition * 2, partition * 3, ordinal + (partition * 2) 
Thread savePlaylist4 = new SavePlaylist(this, partition * 3, media.size(), ordinal + (partition * 3) 
savePlaylistl. startO; 
savePlayl ist2 .startO; 
savePlaylist3 .startO; 
savePlayl ist4 .startO; 
saveNews.startO; 
saveAds.startO; 
saveTips.startO; 

deleteHighOrdinals(conn, highOrdinal - 1); 
// everybody done yet? 
saveOrigins(conn); 
try 
{ 



saveNews.joinO; 
saveAdsjoinO; 
saveTipsjoinO; 
savePlaylistl joinO; 
savePlayl ist2join0; 
savePlaylist3joinO; 
savePlaylist4join(); 

} 

catch (InterruptedException e) 
{ 

Util.debug("Playlist::save was interrupted while waiting"); 

} 

UtiLprintElapsedTimeC'SavePlaylist", startDate); 
return true; 



*/ 



private void saveClips(DBConnection conn, Vector clips, String storedProc) 
{ 

for (int i = 0; \ < clips.size(); i++) 
{ 

Clip aClip = (Clip) clips.elementAt(i); 
String sql = "exec " + storedProc + " " 
+ 1D + "," 

+ aClip.medialD + ", " 
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+ mediaType + n 
+ user ID; 

try 

{ 

DBResultSet rs = conn.executeSQL(sql); 

} 

catch (DBException oops) 
{ 

Util.debug("DB Exception: " + oops.getMessageO); 

} 

} 

} 

public String newLineO 
{ 

return Util.newLine; 

} 

public String toASXO 
{ 

String asx = "<ASX VERSION^-OV PREVIEWMODE^rNOV^ + Util.newLine 

+ UtiLtabO '+ "<REPEAT>" + Util.newLine; 
String streamURL = Constants.STREAM_URL + n ?u=" 

*+ userlD; ~" " • " * ! 1 1 ™ "* - — — ■ « . ■ i . . ■ . ■ . ■ i > . , , . - 1 . >^ ; <.>i : ,%>i*> ^ ,- .» » 1 1 



for (inti = 0;i<10;i++) 
{ • 
asx = asx.concat(Util.tab(2) + 



"<ENTRY>" + Util.newLine 

+ Util.tab(3) 

+ "<REF HREF=V ,W 

+ streamURL 

+ "&n= M 

+ i 

+ M .asp" 

+ "\7> n + Util.newLine 
+ Util.tab(2) 

+ M <tfENTRY>" + Util.newLine); 



} 



} 

asx « asx.concat(Util.tabO + w <yREPEA1>" +Util.newLine 

+ "</ASX>" ■+ Util.newLine); 

return asx; 



} 

Playlistjava Page 10 of I 0 1 1/05/99 1 :38 PM 
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Playlist2 

package com. launch. PlaylistGenerator; 
import java.util.*; 

//- 

/** 

* @author Ted Leung 

* ©version 1999-09-22 
**/ 

//- — 

public final class Playlist2 implements Java. io. Serializable 
{ 

U******* *************************************** ************************ 

II variables 

H******************************************** ******************** ****** 

/** all these vectors contain exclusively Strings which are directory/filename of audio files */ 
public Vector songs; 
public Vector news; 
public Vector ads; 
public Vector tips; 

H********************************************************************** 

// methods — - - : - * •- - - • - •• • - ^ 

II********************************************************************** 

public Playiist20 
{ 

songs = new Vector(50); 
news = new Vector(10); 
ads =new Vector(10); 
tips =new Vector(10); 

} 

II 

/** 

//- — 

public final String toStringO 
{ 

return 
( 

H songs== M +songs.toStringO + M + 
M news= M +news.toStringO + V + 
"ads="+ads.toStringO + M + 
"tips="+tips.toStringO 

); 

} 

II********************************************************************** 

) 

Playlist2.java Page 2 of 2 1 1/05/99 1 :28 PM 
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PlaylistCreatorTest 

package com.launch.PlaylistGenerator; 
public class PlaylistCreatorTest 
{ 

public static void main(String[] args) 
{ 

Util.debugCusing database server " + Constants. DB__SERVER); 

SonglnfoCache songCache = new SonglnfoCache(null); 
songCache.ratingsCache = new RatingsCacheO; 
// Play listParameters params = new PlaylistParameters(377 1 , null, 0, 1 3302); 

PlaylistParameters params = new PlaylistParameters(6474126, null, 0, 6474126); 
PlaylistGenerator gen = new PlaylistGeneratortparams, songCache, null); 
Playlist playlist = gen.create(true, null); 

gen.toMatrix(null,Uti!.DlSPLAY_TEXT); 
System.exit(0); 

} 

PlaylistCreatorTest.java Page 1 of 1 1 1/05/99 1 :35 PM 
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PlaylistEntry 

package com.launch.PlaylistGenerator; 
import java.io.*; 

public class PlaylistEntry implements Serializable 
{ 

public String title, filepath, songTitle, albumTitle, artistTitle; 
public int medialD, songID, albumID, artistID; 

public short implicit; 
public byte origin; 

} 

Play listEntry.java Page 1 of 1 1 1/05/99 1 :28 PM 
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PlaylistGenerator 

package com. launch. PlaylistGenerator; 

import java.util. Vector; 

import java.util. Date; 

import javax.servletServIetOutputStream; 

import java.util. Enumeration; 

public class PlaylistGenerator 

{ 

public final static byte RATER JDJ « 1 ; 
public final static byte RATER_BDS = 2; 
public final static byte RATER_G EN RE = 3; 

private short factor = (short)Constants.DEFAULT_PICK_FACTOR; 

private short ratio = (short) Constants.DEFAULTJJNRATEDJtATIO; 

private int playlistSize = Constants.DEFAULT_PLAYLIST_SlZE; 

private int playlistID; 

private boolean haveTitles = false; 

private Date startDate; 

private Date lastDate; 

private int userlD; 

■ private int djID; w «^^^r«^^ — ***** — „ — — — 

private int mood ID; 

private short mediaType; — 

private IntHash ratings; 
private ItemsProfile items; 
private PlayDates lastPlayed; 
private Population songs; 
private Vector news; 
private Vector ads; 
private Vector tips; 
private DJList djs; 
private GenreList genres; 
private Bandwidth speed; 
private MediaFormat format; 

private StationList stations; 
private ServletOutputStream out; 
private SonglnfoCache songCache; 
private boolean playExplicitLyrics = true; 
/»* 

* Creates a new play list generator. 
*/ 

public PlaylistGeneratorQ 
{ 

songs =newPopulatioriO; 
news =newVectorO; 
ads =newVector(); 
tips = new VectorO; 
ratings = new IntHashO; 
djs =newDJListO; 
items == new ItemsProfiieO; 
lastPlayed = new PlayDatesO; 
genres - new GenreListO; 
stations - new StationListO; 

} 

public PlaylistGenerator (PlaylistParameters params, SonglnfoCache cache, ServletOutputStream out) 
{ 

thisO; 

userlD = params.userlD; 
moodID = params.moodID; 
djlD = params .dj ID; 
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if (djID <= 0) djID = userlD; 

speed = params. speed; 
format = params. format; 

playlistSize - params.pIaylistSize; 
songCache = cache; 
this.out = out; 

} 

private void getRandomO 
{ 

Date startDate = new DateO; 

Song ditty; 

SongData data; 

Songlnfo info; 

SongList songList; 

int rowCount = 0; 

double pickCount; 

double totalSongs; 

// the simple way 

/* 

songList = cache.getInGenres(genres); ^ — — — r - ^ — r ... - - ^ — 

pickCount = Mam.minCsongList.sizeO, mis.RANDOM_SONGS_COUNT); 

// import them all 

if (pickCount = songListsizeO) 

{ 

for (int i = 0; i < pickCount; i++) 
{ 

info = songList.e!ementAt(i); 

rowCount += addRandom(info, SongData SOURCE_RANDOM); 

} 

} 

// import a random subset 

else 

{ 

for (int i = 0; i < pickCount; i++) 
{ 

info = songList.pickRandom(); 

rowCount += addRandom(info, SongData.SOURCE_RANDOM); 

} 

} 

*/ 

// the faster(?) but way more complicated way 

intsongCount = songCache.countInGenres(genres); 

totalSongs = £ songCache.size(SongInfoCache.TYPE_SONG); 

double percent = (songCount / totalSongs) * 100.0; 

UtiLprintElapsedTime( , 'GetRandom done counting in genres**, startDate); 

// the problem is if we pick randomly and they want songs from 

// only a few genres, we're probably not going to get enough to create 

// a playlist So instead, if there's not a whole lot of songs in those genres, 

//just get them directly from the genres instead of taking our chances with random 

Util.debug("getRandom: " + songCount + " non-unique songs in genres (" + percent + "%)"); 

if (percent < Constants.MlN_SONGSJN_GENRES_TO_GET_RANDOM) 

{ 

Util.debug( M getRandom: getting directly from genres"); 
// get the list of songs from each genre 

// choose the number to pick from each, proportional to the number of songs 
// pick them 

int totalToPick ~ Math.min(Constants.R ANDOM_SONGS_COUNT, songCount); 

for (int i = 0; i < genres.sizeO; i++) 

{ 
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songList = songCache.getInGenre(genres.genreAt(i)); 
pickCount = totalToPick * (songList.size() / ((double) songCount)); 
for (int j = 0; j < pickCount; j-H-) 
{ 

info - songListpickRandom(); 
if (info null) 

{ 

rowCount += addRandom(info, 

SongData.SOURCE_GENRES); 

} 

} 

} 

} 

else 
{ 

Util.debugC'getRandom: picking randomly from all songs"); 
for (int i = 0; i < Constants.RANDOM_SONGS_COUNT; i++) 
{ 

// this is really fast 
info = songCache.randomSongO; 
// this is really slow 

— - — —< ■ — rowCount += addRandom(info, SongData.SOURCE -RANDOM); 

} . .. . 

} 

Util.debugC'getRandom added " + rowCount + " songs"); 
Util.printElapsedTime( n GetRandom done", startDate); 

} 

private int addRandom(SongInfo info, byte source) 

{ 

SongData data = songs.initSongGetData(info.songID, Song.UNRATED); 
if (data != null) 
{ 

data.querySource - source; 
data.setInfo(info); 
return 1; 

} 

return 0; 

} 

private void getPopular(SongList list) 
{ 

Date startDate = new DateO; 
Song ditty; 
SongData data; 
Songlnfo info; 
int rowCount = 0; 
if (list != null) 
{ 

for (int i = 0; i < list.sizeO; 
{ 

info = listelementAt(i); 
data = songs.getSongData(info.songID); 
if(data!=null) 
{ 

// we can't add it, but lefs append the info while we're here 
data.setlnfo(info); 

} 

else 
{ 

data = songs.initSongGetData(info.songID, Song.UNRATED); 

if (data != null) 

{ 

data.querySource = data.SOURCE_POPULAR; 
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data.setlnfo(info); 

} 

rowCount++; 

} 

} 

} 

Util.debug("getPopular added " + rowCount + " songs"); 

} 

/*♦ 

* Gets all the required media and data to generate a play list. 
V 

private void gatherMedia(DBConnection conn) 
{ 

Thread getLastPlayed = new GetLastPlayed(lastPlayed, userlD, out); 

Util.out(out, "starting gathering threads at " + timeStampO); 

// try to start them in ascending order of speed 

getLastPlayed.startO; 

// get djs, genres, and bds subscriptions 

getSubscriptionsfconn, djlD, moodlD); 

UtH.out(out, "getSubscriptiohs done ■" + timeStampO); 

// we need to wait for die djs to come in first 

— ~~~Thread getRatings - newGetRatings(songsritemsrdjIDrdjs~son^^^ — — 

getRatings.startO; . \ r . . ; „ 

Util.out(out, "AH threads started n + timeStampO);' 

// getpopular and getrandom should not be threads since they are purely processor bound now 

getPopular(songCache.getP6pular(mediaType)); 

Util.out(out, "getPopular done " + timeStampO); 

getRandomO; 

Util.out(out, "getRandom done (picked " + Constants.RANDOM_SONGS_COUNT + " songs) 

timeStampO); 

UtiI.out(out, "genres for mood " + moodlD + ":" + genres.toStringO); 
// wait for them to finish 

{ 

getRatings.joinO; 
getLastPlayed.joinO; 

catch (InterruptedException oops) 
{ 

UtiLdebugC'InterruptedException: " + oops.toStringO); 

} 

Util.out(out, "gatherMedia done " + timeStampO); 

} 

public void getSubscriptions(DBConnection conn, int userlD, int moodlD) 
{ 

Date started = new DateO; 

try 

{ 

DBResuItSet rs = conn.executeSQL( M exec spJcoGetAllSubscriptions_xsxx " 

+ userlD + '\ 
+ moodlD); 

int raterlD; 
int raterType; 

while (!rs.getBOF0 && Irs.getEOFO) 
{ 

raterlD = rs.getInt( ,, raterID ,, ); 
raterType - rs.getInt( n raterType M ); 
if (raterType — RATER DJ) 

{ 

djs.addElement(new DJ(raterID)); 

} 

else if (raterType = RATERJ3ENRE) 
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{ 

genres.add((short) raterlD); 

} 

else if (raterType = RATER_BDS) 
{ 

stations.addEIement(new Station(raterlD)); 

} 

rs.nextO; 

} 

Util.debug("getSubscriptions added n 

+ djs.sizeO + " DJs, " 

+ genres.size() + " Genres, n 

+ stations.sizeO + " Stations"); 

} 

catch (DBException oops) 
{ 

UtiLdebug( n DB Exception in getSubscriptions " + oops.getMessageO); 

} 

Util.printElapsedTimeO'getSubscriptions took started); 

Calculates scores for all the songs and puts them into the various vectors 

public void processSongsO 
{ 

byte result; 

WeightMatrix weights = new WeightMatrixO; 
Integer songID; 
SongaSong; 
SongData data; 
short type; 
Date playedAt; 
Songlnfo info; 
int good = 0; 
int tested = 0; 
int artistlD, albumID; 
Item albumltem; 
Item artistltem; 

AlbumArtistData albumAndArtist = new AIbumArtistData(); 

IntHash reasons = new IntHashO; 
double now = lastPIayed.dbDate.getTimeO; 
double lastThreeHours = Util.MILLISECONDS_rN_SECOND ♦ 

Util.SECONDS_IN_MlNUTE * 
UtiI.MINUTES_IN_HOUR * 

3; 

for (Enumeration e = songs.keysQ; e.hasMoreElementsO ;) 
{ 

tested++; 

albumAndArtistresetO; 

songID = (Integer) cnextEIementO; 
aSong = songs.get(songlD); 
data = aSong.getDataO; 

if (aSong.getTypeO = Song.EXCLUDED) 
{ 

reasons. increment 1 ); 

> 
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// add the song info 

info = data.getInfo(); 

// get the song info from the cache 

if (info = null) 

{ 

info = (Songlnfo) songCache.get(songID, 
data.setlnfo(info); 

} 

// if it's still null, it's not encoded 

if (info = null) 

{ 

aSong.setType(Song.EXCLU DED); 

reasons. increment(2); 

continue; 

} 

// ok, we have the song info, 
//add last played 

playedAt = lastPlayed.get(songID); 

Mf (playedAt != null) — — "~ ~ 

. ( . . 
lastPIayed.remove(songID); 

// don't play the same song twice in a 3 hour period 
if (now - playedAtgetTimeO < lastThreeHours) 
{ 

// mark songs played in the last three hours 

// so as to comply with the RI AA rules 

// and make sure we don't pick too many later 

artistID- data;getArtistID0; 

albumID = data.getAlbumID{); 

// "various artists" albums don't count 

if (!ArtistInfo.isVariousArtists(artistID)) 

{ 

songs.artistCounts. increment(artistl D); 

} 

songs.albumCountsJncrement(albumIE>); 

// make sure we dont play this again so soon 

aSong.setType(Song.EXCLUDED); 

reasons.increment(3); 

continue; 

} 

data.lastPlayed = lastPlayed.getScore(songlD); 

} 

// check for bad words 

if (IplayExplicitLyrics && info.hasExplicitLyricsO) 
{ 

aSong.setType(Song.EXCLUDED); 

reasons. increment(4); 

continue; 

} 

// now check for media in the type we need 
if (!info.media.inType(mediaType)) 
{ 

aSong.setType(Song.EXCLUDED); 

reasoiis.increment(5); 

continue; 

} 

// check for valid genres 

if (! info.album.inGenres(genres)) 
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{ 

// for popular songs, don't exclude them, 

// otherwise we won't be able to default to them 

// if the genre restrictions are too tight 

if (data.querySource = data.SOURCE_POPULAR) 

{ 

songs. remove(songlD); 

} 

reasons.increment(6); 

aSong.setType(Song.EXCLUDED); 

continue; 

} 

// we got this far, so try to calculate an implicit rating 
result = data.calculatelmplicit(items, album And Artist); 
if (result — SongData.EXCLUDEJtfE) 
{ 

aSong.setType(Song.EXCLUDED); 

reasons. increment(7); 

continue; 

} 

if (result = SongData.MAKE_ME_IMPLICIT) 

aSong.setType(Song IMPLICIT); 
data.calculateDJs(items, albumAndArtist); 
data.score(weights, stations); 
songs, imp iicitaddElement(data); 
good++; 



} 

else 
{ 



type = aSong.getTypeO; 

// put the song in a list to pick from later 

if (type = Song.EXPLICIT) 

{ 

// your djs don't matter if you explicitly rated the song 
songs.explicit.addElement(data); 

} 

else if (type = Song.IMPLICIT) 
{ 

data.calculateDJs(items, albumAndArtist); 
songs. implicit.addEtement(data); 

} 

else if (type = Song.UNRATED) 
{ 

data.calcu!ateDJs(items, albumAndArtist); 
songs.unrated.addElement(data); 

} 

// calculate the score 
data.score(weights, stations); 
good++; 



} 

Util.out(out, "scores calculated n + timeStampO); 

// for all the songs we didnt get for whatever reason, make sure we 
// are accounting for their plays for compliance with R1AA rules 
lastPlayed.markRecentIyPIayed(songCache, songs); 
Util.out(out, "recently played albums and artists marked H + timeStampO); 

Util.out(out, "Of rt + tested + " songs, these are the reasons for exclusion: " 
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+ reasons.get(l) + " were already excluded, " 
+ reasons.get(2) + " were not encoded, " 
+ reasons.get(3) + " were played in the last 3 hours, " 
+ reasons.get(4) + " had explicit lyrics, " 
+ reasons.get(S) + " were not in mediaType " + mediaType + ", M 
+ reasons.get(6) + *' were not in their genres, " 
+ reasons. get(7) + " had an implicit rating of 0.")i 
Util.out(out, "There are " + good + " songs available for play"); 

} 

/*♦ 

* Gets a user's preferences for their play lists 
*/ 

public boolean getOptions(DBConnection conn) 
{ 

int rowCount = 0; 

short tempRatio; 

short bandwidth = 0; 

// returns: ratio, factor, mediaType 

String sql = "exec sp_IcGetPreferences_xsxx " + userlD; 

{ 

„ — ^DBResultSet rs = conn.executeSQL(sql); 

if (!rs.getBOF0 && !rs.getEOF0) 

terapRatio= (short) rs.getInt("unratedQuota"); 
if (tempRatio > 0 && tempRatio < 1 00) 

ratio = tempRatio; 
playExplicitLyrics = rs.getBooIean("expIicit"); 
// if there was no mediatype set from the parameters 
// set it to the default 
if(!speed.isSetO) 

speed.set(rs.getShort( M bandwidth")); 

rowCount++; 

} . 

} 

catch (DBException oops) 
{ 

UtiI.debug("DB Exception in getOptions: " + oops.getMessageO); 

} 

mediaType = Media. getMediaType(speed, format); 
Util.debug("PIay dirty songs?: n + playExplicitLyrics); 
Util.debug("Bandwidth: " + speed.toStringO); 
Util.debug("Format: " + format.toStringO); 
Util.debug("mediaType: " + mediaType); 
return (rowCount > 0); 

} 

/** 

♦Creates a playlist. 
*/ 

public Playlist createPlaylist(DBConnection conn) 
{ 

Util.out(out, "start of createPlaylist " + timeStampO); 
Playlist playlist = new Playlist(pIayIistID); 
gatherMedia(conn); 
processSongsO; 

playlist = makePlaylist(factor, ratio, playlistSize, playlist); 
Util.out(out, "end of createPlaylist M + timeStampO); 
return playlist; 

private void logCreate(DBConnection conn) 
{ 
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try 
{ 

conn.executeSQL("exec splcLogP lay list _jxxx " 

+ userlD + " 
+ djID + w 
+ moodID + ", " 
+ 0 + H 

+ mediaType + " 
+ elapsedTimeO 

); 

} 

catch (DB Exception e) 
{ 

Util.debug("DBException in logCreate: M + e.toStringO); 

} 

} 

/** 

♦ Creates and immediately saves a playlist. 
»/ 

public Playlist create(boolean save, SimplePlaylist oldPlaylist) 
{ 

DBCbnnection conn -null; ~ — *" " - ™ ' ^^^^ 

Playlist playlist = null; 
try 
{ 

conn = new DBConnectionO; 
getOptions(conn); 
playlist - createPlaylist(conn); 
UtiLout(out, "starting to save playlist " + timeStampO); 
if (save) 

playlist.save(conn, oldPIaylist); 
logCreate(conn); 

UtiLout(out, "done saving playlist " + timeStampO); 
conn.closeO; 

} 

catch (DB Exception oops) 
{ 

Util.out(out, "DBException in create: " + oops.getMessageO); 

} 

catch (Throwable e) 
{ 

System.err.println( ,, Generic Exception caught in PlaylistGenerator: " + e.toStringO); 
e.printStackTraceO; 

} 

return playlist; 

} 

public Playlist makePlaylist(int factor, int ratio, int playlistSize, Playlist playlist) 
{ 

Util.out(out, "ordering..." + timeStampO); 
songs.sort(songs. explicit); 
songs.sort(songs.implicit); 
songs.sort(songs.unrated); 

Util.out(out, "finished sorting vectors at " + timeStampO); 
playlistxounts = new PickCount(userID, djID, ratio, playlistSize, songs, out); 
// set up the playlist 
playlistuserlD = this.userlD; 
playlist.moodID = this.moodID; 
playlist.djID = this.djID; 
playlist.mediaType =» this.mediaType; 
// copy the list of albums and artists recently played 
//for the RIAA rules 

playlistalbums = (IntHash) songs.albumCountsxIoneO; 
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playlist.artists = (IntHash) songs.artistCounts.clone(); 

// pick songs 

pickSongs(playlist); 

// check if we got everything we need 

if (playlist.media.size() < playlistSize) 

{ 

L)til.out(out, "We only got " + playlist.media.sizeO + " songs for user " + playlist.userlD 
+ Playing popular music in mediaType " + mediaType); 

// uh oh, we didnt get enough songs; play popular stuff 

playlist.counts.explicit = 0; 

playlistxounts.implicit = 0; 

play list.counts. unrated = playlistSize; 

playlist.albums = (IntHash) songs. albumCounts.cIoneO; 

playlist.artists = (IntHash) songs.artistCountsxlone(); 

playlist.resetSourcesO; 

playlist media.removeAHElementsO; 

playlistpopularOnly = true; 

songs.importPopular(songdache.getPopular(rnediaType), lastPlayed, playExplicitLyrics); 
pickSongs(playlist); 

} ■" _ ____ 

" '""7/ pick news ™ • * : T - - - --■ - 

pickNews(pIaylist); ; / .^i^c:^^- 

Util.out(out, "picked news " + timeStampO); 
// pick ads 
pickAds(pIaylist); 

*Util.out(out, "picked ads " + timeStampO); 
// pick tips 
pickTips(playlist); 

Util.out(out, "picked tips " + timeStampO); 
Util.out(out, "playlist has " + playlistlengthO + " songs"); 
UtiLout(out, "shuffling play list.."); 
return playlistshuffleO; 

} 

public void pickNews(PlayIist list) 
{ 

list.news = songCache.randomGlipList(SongInfoCache.TYPE_NEWS, mediaType, 
Constants.MAX_NEWS_ITEMS); 
) 

public void pickAds(PIaylist list) 
< 

list ads = songCache.randomClipList(SongInfoCache.TYPE_AD, mediaType, 
Constants.MAX_ADS); 
} 

public void pickTips(Play!ist list) 
{ 

Iisttips = songCache.randomClipList(SongInfoCache.TYPE_TIP, mediaType, 
Constants.MAX_TIPSJTEMS); 
} 

public Playlist pickSongs (Playlist list) 
{ 

Util.out(out, "start of pickSongs " + timeStampO); 
PickList pickTypes = new PickList(list.counts); 
int pickOrder = 0; 
int iteration = 0; 

int artistID, albumID, artistCount, albumCount; 

short type; 

SongData pick; 

SongGroup songGroup; 

while (pickTypes.sizeO > 0) 

{ 

iteration++; 
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pick = null; 

songGroup = null; 

// get a group to pick from 

type = pickTypes.getRandom(); 

if (type = Song.EXPLICIT && songs.explicit.sizeO > 0) 
{ 

songGroup = songs. explicit; 

} 

else if (type — Song.lMPLlCIT && songs.implicit.sizeO > 0) 
{ 

songGroup = songs, implicit; 

} 

else 
{ 

type = Song. UNRATED; 
songGroup = songs.unrated; 

} 

// pick a random song from a group 
pick = songGroup.pickRandom(factor); 
// if we have hone of that type, try another 
if(pick = null) 

pickTypes.reAdd(type, songGroup, songs); 
~ * continue; - .-. . ~ . --v ■ . ..... 

} 

artistID = pick.getArtistlDO; 

albumID = pick.getAlbumID(); 

artistCount = 0; 

albumCount =0; 

// check for RJ A A compliance 

// various artists and soundtracks don't count 

if (!ArtistInfo.isVariousAitists(artistID)) 

artistCount = list.artists.get(artistID); 
albumCount = listalbums.get(albumID); 
if (artistCount >= Constants.RlAA_MAX_SONGS_BY_ARTIST 

|| albumCount >= Constants.RIAA_MAX_SONGS_FROM_ALBUM) 

{ 

pick.status.status ~ PickStatus.REJECTED; 

//Util.debug("Song rejected by RIAA"); 

// we have too many from this artist or album. Try again. 

pickTypes.reAdd(type, songGroup, songs); 

continue; 

// increment the album and artist counts 
if(!ArtistInfo.isYariousArtists(artistID)) 

list.artists.increment(artistID); 
list.albums. increment(aIbumID); 
// add it to the playlist 
list.append(pick); 

pick.status.status = PickStatus.PICKED; 
pidcstatus.order = ++pickOrder; 

} 

songs.ordered = false; 

Util.out(out, "end of pickSongs " + timeStampO); 
return list; 

} 

public void toMatrix(ServIetOutputStream out, int displayType) 
{ 

songs.orderO; 
String hi begin = ""; 
String hlend = m '; 

if (displayType = Util.DISPLAY_HTML) 
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{ 

hlbegin = "<P><H1>"; 
hlend = "</Hl>"; 

} 

Util.out(out, hi begin + "Item Ratings" + hi end + UtiLnewLine); 
items.print(out, songCache); 

Util.out(out, hlbegin + "Explicitly Rated Songs" + hlend + UtiLnewLine); 

songs.toMatrix(out, Song.EXPLICIT, dispIayType); 

Util.out(out, hlbegin + "Implicitly Rated Songs" + hlend + UtiLnewLine); 

songs.toMatrix(out, Song. IMPLICIT, dispIayType); 

Util.out(out, hlbegin + "Unrated Songs" + hlend + UtiLnewLine); 

songs.toMatrix(out, Song. UNRATED, dispIayType); 
// + h 1 begin + "Excluded Songs" + hlend + UtiLnewLine 

// + songs.excludedList(); 

} 

public String timeStampO 
{ 

Date now = new DateO; 
if(startDate = nul!) 

{ . 
startDate - lastDate — now; 

. _ double diff= (now.getTimeO - lastDate.getTimeO)/ 1000.0; 

double total = (now.getTimeO - startDate.getTimeO) / 1000.0; 
lastDate = now; 
return UtiLnewLine 

+ « » + UtiLnewLine 

+ diff + " lap time, " + total + w total" + UtiLnewLine 

+ " » + UtiLnewLine; 

} 

public double elapsedTimeO 
{ 

Date now = new DateO; 
if(startDate = null) 
{ 

startDate = lastDate - now; 

} 

return (now.getTimeO - StartDate.getTimeO) / 1000.0; 

} 

} 
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PlaylistGeneratorServIet 

package com . launch. P lay I istGenerator; 
import Java. io.*; 

import javax.servlet.http.HttpServlet; 
import javax.servlet.http.HttpServIetRequest; 
import javax.servIet.http.HttpServletResponse; 
import j a vax.servlet.ServletConfig; 
import javax.servlet.ServletException; 
import javax.servlet.ServletOutputStream; 
import java.util.*; 
/** 



* PlaylistGeneratorServlet.java 6/30/99 

* Servlet that creates LAUNCHcast playlists 

* Copyright (c) 1999 Launch, Inc. 

* @author Jeff Boulter 

* ; ; 

public class PlaylistGeneratorServIet extends HttpServlet { 

SonglnfoCache songCache; 
Thread cacheUpdater; 

public void generatePlaylist(HttpServIetRequest request, 

HttpServletResponse response) throws lOException 

< 

// get stream for output 

ServletOutputStream out = response.getOutputStreamO; 
GeneratorParameters prop = new GeneratorParameters(request); 
if (prop.debugO) 

response.setCoritentType("text/plain"); 

else 

response.serContentTypeCvideo/x-ms-asf'); 

PlaylistParameters params ~ new PlaylistParameters(prop); 
P lay HstStatus status - new Play listStatus(prop.userIDO); 
status.init(out); 
if (prop.debugO) 

out.print(status.toStringO); 

boolean generate = true; 

// no need to regenerate right now, use an old play list 
if (prop.forceRefreshO) 

{ 

if (prop.debugO) outprintln("generating because forceRefresh is on"); 

} 

else if (status. tsStaleO) 
{ 

if (prop.debugO) out.println( M generating because the playlist is stale"); 

} 

else if (prop.speedO.isSetO && (prop.speed0.get() != status.speed.get())) 

{ 

if (prop.debugO) out.println("generating because the mediaTypes are different"); 

} 

else if (prop.formatO.isSetO && (prop.formatO-getO !- status.format.getO)) 

{ 

if (prop.debugO) out.println( H generating because the media formats are different"); 
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} 

else if (prop.moodID() != status, mood ID) 

{ 

if (prop.debugO) out.println("generating because the moods are different"); 

} 

else if (prop.djID() != status.djID) 
{ 

if (prop.debugO) out.println("generating because the djs are different"); 

} 

else 

generate - false; 

if ((generate) // we can use an old play list 
{ 

// reset the ad, news, and tip dates 

if (status.playlist != null) 
{ 

status.resetDates(); 

■ } 



"<PRE>"); 



Play list play list = new PlaylistO; 
playlistuserlD = status.userlD; 

out.print(playlist.toASXO); 



} 

else // we have to generate the playlist 
{ 

ServIetOutputStream outStream = null; 

if (prop.debugO) 
{ 

outStream = out; 

outprintln("regenerating playlist with parameters: " + params.toStringO + 

outflushO; 
} 

PlaylistGenerator gen = new PlaylistGenerator(params, songCache, outStream); 
Playlist playlist = gen.create(!prop.dontsaveO, null); 

if (prop.debugO) 
{ 

out.println("</PRE>"); 

if (prop.debugFormatO = Util.DISPLAY_TEXT) 
out.printin( K <PRE> M ); 

out.printIn(playIisttoStringO 

+ "<P>"); 

if (prop.matrixO) 
{ 

// out.printIn( n <FONT SIZE=-l> n ); 

gen.toMatrix(out, prop.debugFormatO); 
// out.println( ,, <yFON1>"); 

} 

if (prop.debugFormatO — UtiLDISPLAY TEXT) 

out.println("</PRE>"); 
out.println("<XMP> n + playlist.toASXQ + n </XMP>"); 

} 
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else 

out.print(playlist.toASXO); 

} 

outcIoseO; 

} 

public void refreshPlaylist(HttpServIetRequest request, 

HttpServletResponse response) throws IOException 

{ 

// get stream for output 

ServIetOutputStream out = response.getOutputStreamO; 

response.setContentType( M text/plain n ); 

// this is the stuff coming in on the query string 

GeneratorParameters prop - new GeneratorParameters(request); 

PlaylistParameters params = new PlaylistParameters(prop); 

// this is what's in their current playlist 

PlaylistStatus status = new PlaylistStatus(prop.userIDO); 
status.init(out); 



.. . ■ . if (prop.debugO) 

outprint(status.toStringO); 

if(status.isStaleO) 
{ 

ServIetOutputStream outStream = null; 

params = new PlaylistParameters(status); 

if (prop.debugO) 

{ 

outStream = out; 

outprintIn( M refreshing playlist with parameters: " + params.toStringO); 
outflushQ; 

} 

PlaylistGenerator gen - new PIaylistGenerator(params, songCache, outStream); 
Playlist playlist = gen. create{! prop. dontsaveO, status.playlist); 

} 

else 
{ 

out.println("No need to refresh playlist now"); 

} 

outcloseQ; 

} 

public void doGet ( 

HttpServIetRequest request, 

HttpServletResponse response 

) throws ServIetException, IOException { 

try 

{ 

//Utii.debug("PlaylistGeneratorServlet recieved a Get"); 
// prevent caching 

response.setHeader("Pragma", "no-cache"); 
response.setHeader("Cache-contror\ "no-cache"); 
response.setHeade^Expires", "0"); 
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// figure out what we need to do 

String actionStr = requestgetParameterC'action"); 

if (actionStr = null) 

actionStr = new StringC generate"); 
if (actionStr.equals("refresh M )) 

{ 

refresh Play I ist(request, response); 

else if (actionStr.equals( M cachestatus")) 
{ 

ServletOutputStream out = response.getOutputStreamO; 
response.setContentType("text/plain M ); 

songCache.ratingsCache.status(out, request getParameter("detai!") != null); 
out.close(); 

} 

else //default action 
{ 

generatePlaylist(request, response); 

> 

} 

' ^-^ catch (Throwablee) * ' * - — — - — 

{ . • . ■ ... , 

System.err.println(new Date0.toStringO + " Caught an exception in doGet: " + 

e.toStringO); 

e.printStackTraceO; 

public void doPost(HttpServletRequest req, HttpServlet Response resp) throws ServletException, 
lOException 
{ 

UtiLdebugCPIaylistGeneratorServlet recieved a Post"); 

try 

{ 

String use^agent^eq.getHeaderC'USER^AGENT"); 

if(user_agent.equaIs(com.launch.misc.constants.PLAYLIST_SERVER)) 
{ 

// need to generate play list and return it 
GeneratorParameters prop = new GeneratorParameters(req); 
PlaylistParameters params = new PlaylistParameters(prop); 
PlaylistGenerator gen ~ new PlaylistGenerator(params, songCache, null); 
Playlist playlist = gen.create(true, null); 

Playlist2 playlist2 =playlist.toPlaylist2(); 

ObjectOutputStream oos=new ObjectOutputStream(resp.getOutputStreamO); 

oos.writeObject(playlist2); 

oos.flushO; 

oos.cIoseQ; 

} 

else if (user_agent.equals(com.Iaunch.misc.constants.RATING_WIDGET)) 
{ 

// need to update cache with new info 

int data_size=req.getContentLength(); 
byte bO=new byte[data_size]; 
req.getInputStreamO-read(b,0,data_size); 
Vector v=(VectorXnew ObjectInputStream(new 
ByteArrayInputStream(b))).readObject0; 

Util.debug("received a list of changed ratings " + v); 
// need to tell cache of these changes 
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Enumeration e=v.elements(); 
while (e.hasMoreElementsO) 
{ 

songCache.ratingsCache.put!ntoCache((CachedRating)e.nextElementO); 
} 

} 

else 
{ 

System.err.println("PlaylistGeneratorServlet received a post from an unknown 

person : n + user_agent); 

> 

catch (Throwable t) 
{ 

t.printStackTraceO; 

} 

} 

/** 

* Initialization method - 
* 

public void init (ServletConfig conflg) throws ServIetException . . — .. . ~ ; "\ r ' -'- — 

{ " ' '■ " " *" ~ 

super.init(config); 

songCache = new SongInfoCache(null); 
// start the updater thread 

cacheUpdater = new SongInfoCacheUpdater(this); 

cacheUpdater.set?riority(Thread.MIN_PRJORITY); 

cacheUpdater.startO; 

songCache.ratingsCache = new RatingsCacheO; 

} 

/** 

* Destroy method - 

* get rid of the api 

* servlets "should have" a destroy method for garbage collection 
V 

public void destroyO 
{ 

cacheUpdater.stopO; 
cacheUpdater = null; 
songCache -null; 

} 

} 

PlaylistGeneratorServletjava Page 5 of 5 11/05/99 1:21 PM 
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PlaylistMaker 

package com. launch. Play listGenerator; 
import javax.servlet.ServletOutputStream; 
/** 

* this is the dumb class for ASP 
*/ 

public class PlaylistMaker 
{ 

public PlaylistGenerator generator; 
public Playlist playlist; 

public PlaylistMakerO 

{ 

generator = new PlaylistGeneratorQ; 

} 

public void init(int userlD, int djID, short mediaType, int moodID, int playlistID) 

■••{'" 

// generator.init(userID, djID, moodID); 

— ■ — — 

public int make() 

playlist = generator.create(faIse, null); 

return playlist.ID; 

} 

public int makeAndSaveO 
{ 

playlist - generator.create(true, null); 
return playlist.ID; 

} 

public void toMatrix(ServIetOutputStream out, int displayType) 
{ 

generator.toMatrix(out, displayType); 

} 

public String toASX() 
{ 

return play!ist.toASXO; 

} 

} 

PlaylistMaker.java Page I of 1 1 1/05/99 1 :32 PM 
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PlaylistParameters 

package com.launch.PIaylistGenerator; 

public class PlaylistParameters 

{ 

int userlD; 
int dj!D; 

int playlistSize = Constants. D EFA U LT_PL AY LI STS1ZE; 
int mood ID; 

Bandwidth speed = new BandwidthO; 
MediaFormat format ~ new MediaFormat(); 
public PlaylistParameters(int userlD) 

{ • 

this.userlD = djID = userlD; 

} 

public PlaylistParameters(int userlD, Bandwidth speed, int moodID) 
{ 

this(userlD); 

if (speed != null) , 
{ 

this.speed = speed; 

} 

this.moodID = moodID; 

} 

public PIaylistParameters(int userlD, Bandwidth speed, int moodID, int djID) 
{ 

this(userID, speed, moodID); 

if(djID>0) 

this.dj!D = djID; ;.r 

} 

public PlaylistParameters(PIaylistStatus status) 
{ 

this(status.userID, status.speed, status.moodID, status, dj ID); 

> 

public PlaylistParameters(GeneratorParameters prop) 
{ 

this(prop.userIDO, prop.speedO, prop.moodlDO, prop.djIDO); 

} 

public String toStringO 
{ 

return H userID=" + userlD + " 

+ H bandwidth=" + speed.toStringO + n 
+ "moodID=" + moodID + n 
+ M djID=" + djID; 

} 

> ■ 
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PlaylistStatus 

package com. launch. Play listGenerator; 

import java.uti I. Date; 

import javax.servlet.ServletOutputStream; 

public class PlaylistStatus 

{ 

int userlD, newRatingsCount, moodID, djID, songsRemaining; 
short mediaType; 

Date lastPlaylist - new DateO; 

MediaFormat format; 
Bandwidth speed; 

Date dbDate = new Date(); 

public SimplePlaylist play list; 

public PlaylistStatiis(int userlD) 

format = new MediaFonnat(MediaForrnat.WINDOWSMEDIA); 

this.userID = userlD; . , _ - ; 

} 

public String toStringO 
{ 

return "Playlist status for userlD " + userlD + + Util.newLine 

+ " newRatingsCount: " + newRatingsCount + Util.newLine 

+ " moodID: H + mood ID + Util.newLine 

+ djID: " + djID + Util.newLine * 

+ " songsRemaining: " + songsRemaining + Util.newLine 

+ " mediaType: " + mediaType + Util.newLine; 

} 

public void init(ServletOutputStream out) 
{ 

try 
{ 

DBConnection conn = new DBConnection(); 

DBResultSet rs = conn.executeSQL("exec spJcGetPlaylistlnfoForUser xsxx " + 

userlD); 

while (Irs.getBOFO && Irs.getEOFO) 
{ 

newRatingsCount = rs.getlntC'newRatingsCount"); 

lastPlaylist = rs.getTimestamp( l, IastPlaylist"); 

dbDate = rs-getTimestarnpC'dbDate"); 

playlist = SimpIeiPlaylistrTomBytes^.getBytesCplaylist")); 

rs.nextO; 

} 

if (playlist != null) 
{ 

songsRemaining = play!ist.songs.size(); 

moodID = playlist.moodID; 

djID = playlistdjlD; 

mediaType = playlistmediaType; 

speed = Media.typeToBandwidth(mediaType); 

} 
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conn.closeO; 

} 

catch (DBException oops) 

{ 

Util.out(out, "DBException in PlaylistStatus.init: " + oops.toStringO); 

} 



} 



public void resetDatesO 
{ 

if (play list = null) 
return; 

Util.debug(new DateO.toStringO + " Playlist OK, just resetting dates for userlD " + userlD); 

playlistresetI>ates(dbDate); 

playlist.save(userlD); 

} 

public boolean isStaleO 



djlD) 



} 



double one Week - UtiI.MILLISECONDS_IN_SECOND * 

Util.SECONDS_IN_MINUTE * 
Util.MINUTES_IN_HOUR * 
UtiLHOURS JN_DAY ♦ 
UtiLDAYSJNJYEEK; 

if (songsRemaining <= Constants.REFRESH_AT_SONGS_LEFT) 
return true; 

// if you're listening to someone else's station, your new ratings 
// won't make a difference 

if (newRatingsCount>= Constants.REFRESH_AT_NEW_RATINGS_COUNT&& userlD = 
return true; 

if (new DateO-getTimeO - lastPlaylist.getTimeO > one Week) 
return true; 

return false; 



/♦ 

public void flushPIaylist(ServIetOutputStream out) 
{ 

try 
{ 

DBConnection conn = new DBConnectionO; 

DB Results et rs = conn.executeSQL("exec sp_lcFlushPlaylist_xxud " + userlD); 
conn.closeO; 

} 

catch (DBException oops) 
{ 

UtiI.out(out, "DBException in PlayIistStatus::flushPlay!ist: " + oops.toStringO); 

} 

} 

public void deletePlaylist(ServIetOutputStream out) 
try 
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DBConnection conn = new DBConnectionO; 

DBResuItSet rs = conn.executeSQL^exec spJcDeletePlaylistxxud " + userlD); 



conn.close(); 
} 

catch (DB Exception oops) 
{ 

Util.out(out, "DBException in PlayIistStatus::deIetePlaylist: " + oops.toString()); 

} 



DBConnection conn = new DBConnectionO; 

DBResuItSet rs = conn.executeSQLC'exec sp_lcResetClipScheduIe_xxux " + userlD); 
connxloseQ; 



public void resetClipSchedule() 



try 




UtiLdebug("DBException in PIayIistStatus::resetDates: M + oops.toStringQ); 



} 

*/ 
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PopularSongs 

package com. launch. PlaylistGenerator; 
import java.util. Vector; 
import java.uti L Hashtable; 
import java.util. Enumeration; 
public class PopularSongs 

{ 

private Hashtable byMedia; 

public SongList get(short mediaType) 
{ 

return (SongList) byMedia.get(new Short( mediaType)); 

} 

public PopularSongs(Hashtable songs, Hashtable mediaTypes) 
{ 

byMedia == new Hashtable(l); 

// make a list of all songs and sort them 

_ — SongList all = new SongList(songs); '-"- — — 

allsortO; 

// create each of the song lists 

for (Enumeration e = mediaTypes.keysO; e.hasMoreElementsO;) 
{ 

Short mediaType = new Short(((Integer) e.nextElementO) shortValueO); 
byMedia.put(mediaType, new SongListO); 

} 

Songlnfo info; 
Media track; 
SongList list; 

// put each into a separate list for each mediaType 

for (int i = 0; i < all.sizeO; 

{ 

info = all.elementAt(i); 

for (int j = 0; j < info.media.sizeO; j++) 
{ 

track = info.media.typeAt(j); 

list = ((SongList) byMedia.get(new Short(track.mediaType))); 
list.addEIement(info); 

} 

} 

// truncate each list to the top 1 000 most popular songs 

for (Enumeration e = mediaTypes.keysO; e.hasMoreElementsO;) 

{ 

Short mediaType = new Short(((Integer) e.nextElementO) shortValueO); 

list = (SongList) byMedia.get(mediaType); 

listsetSize(IOOO); 

} 



} 

} 
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Population 

package com. launch. PlaylistGenerator; 

import Java. util. Enumeration; 

import java. util. Date; 

import java.text.SimpIeDateFormat; 

import java.util. Vector; 

import java.util.Hashtable; 

import javax.servlet.ServletOutputStream; 

import java.text DateFormat; 

public class Population 

{ 

/♦ 

private int readers = 0; 
private int writersWaiting = 0; 
private boolean writing = false; 
*/ 

private boolean haveTitles — false; 
public boolean ordered =,fcilse; ^ .»■■■ 

public SongGroup explicit; - - ^ - — 
public SongGroup implicit; 
- public SongGroup unrated; 

private Hashtable hash; 

public IntHash artistCounts; 
public IntHash albumCounts; 

public PopulationO 
{ 

explicit = new SongGroupO; 
implicit = new SongGroupO; 
unrated = new SongGroupO; 
artistCounts = new IntHashO; 
albumCounts = new IntHashO; 
hash = new HashtableO; 

} 

/* 

public synchronized void addReaderO 
{ 

++readers; 

} 

public synchronized void removeReaderO 
{ 

—readers; 

if (readers = 0) 

{ 

notifyAIIO; 

} 

} 

public synchronized void requestWriteO 
{ 

++writersWaiting; 

} 
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public synchronized void finish WriteO 
{ 

—writersWaiting; 

if (writersWaiting = 0) 

{ 

notifyAIIO; 

} 

} 

*/ 

// returns this song if it's valid for adding data, null otherwise 

public synchronized Song initSong(int songID, short type) 
{ 

if(type<=0) 

return null; 

boolean result - true; 
/* 

request WriteQ; 



while (readers > 0) 

{ •" ~ : • ' 

try{wait0;} 

catch (InterruptedException e) {} 

} 

writing = true; 
*/ 

Song song = get(songl D); 

if (song = null) 
{ 

song = new Song(songID, type); 
put(songlD, song); 

// if it's excluded, it's not valid for modifying 
if (type = Song-EXCLUDED) 
result = felse; 

} 

else 
{ 

} 



result = song.setType(type); 



if (result) 

return song; 

// writing -false; 

// finishWriteO; 



} 



return null; 



public synchronized SongData initSongGetData(int songID, short type) 
{ 

Song aSpng = initSong(songID, type); 

if (aSong = null) 
return null; 

return aSong.getDataO; 
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} 

public synchronized SongData getSongData(int songID) 

{ 

return getSongData(new lnteger(songlD)); 

} 

public synchronized SongData getSongData(lnteger songID) 

{ 

Song s = get(songlD); 

if(s = null) 

return null; 
return s.getData(); 

} 



public synchronized SongData getSongData(int songID, short type) 
{ 

SongData result = null; 
synchronized (this) 

■• •■" : { ; • : ■ 

while (writers Waiting > 0) 
{ 

try{waitO;} 

catch (InterruptedException e) { } 

} 

addReaderO; 

} 

*/ 

Song song = get(songID); 

// there's no song for that ID; Did you call initSong? 
if (song != null && type >= song.getTypeO) 
result = song.getDataO; 
// removeReaderO; 

return result; 



public synchronized Song get(int songID) 
return get(new Integer(songID)); 



public synchronized Song get(Integer songID) 
return (Song) hash.get(songID); 



public synchronized Song remove(int songID) 
return remove(new Integer(songID)); 



public synchronized Song remove(Integer songID) 

return (Song) hash.remove(songID); 
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} 

private synchronized Song put(int songID, Song song) 
{ 

return (Song) hash.put(new Intege^songlD), song); 

} 

private int availableO 
{ 

int i = 0; 

for (Enumeration e = hash.keys(); e.hasMoreElementsO ;) { 
Song song = get((lnteger) e.nextEIementO); 

if (song.type != Song. EXCLUDED) - 
{ 

i++; 

} 

} 

return i; 

y ■ ' . ... - • ' " - 

public Enumeration keysO - . ... - - :: - z: .-• r ~: . , . -.. . 

return hash.keysO; 

public void orderO 
{ 

createVectorsO; 
sortVectorsO; 

} 

public int excludedCount() 
{ 

int result = 0; 

for (Enumeration e = hash.keysO; e.hasMoreElements() ;) { 

Song song = get(((Integer) e.nextElement()).intVaIueG); 
if (song.type = Song.EXCLUDED) 

{ 

result++; 

} 

} 

return result; 

} 

public boolean isEligible(int songID, int artistID, int albumID) 
{ 

Song song = get(songID); 

if (song != null && song.type = Song.EXCLUDED) 
return false; 

if ((artistCounts.get(artistID) < Constants.RIAA_MAX_SONGS_BY_ARTIST) 

&& (aIbumCounts.get(albumID) < Constants.RIAA_MAX_SONGS_FROM_ALBUM)) 
return true; 

return false; 

} 
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public void create Vectors() 
{ 

explicit.removeAHEIementsO; 
impIicitremoveAllElementsO; 
unrated.removeAHEIementsO; 

for (Enumeration e = hash.keysO; e.hasMoreElementsO;) { 
Util.debug("interation " + i); 
Song mySong = get((Integer) e.nextElementO); 

if (mySong != null) 
{ 

SongData data = mySong.getDataO; 

if (mySong.type = Song. EXPLICIT) 

explicit.addElement(data); 
else if (mySong.type = Song.IMPLICIT) 
impIicit.addElement(data); 

~ = ™ — • - ■ " * else if (mySong.type != Song.EXCLUDED) — — ™~-*- 

■. . . unrated.addElement(data); 

} 

} 

public void importPopuIar(SongList abunch, PlayDates lastPlayed, boolean playBad Words) 
{ 

Songlnfo info; 
SongData data; 
Song ditty; 
int added = 0; 

unrated.setSize(0); 

long now = new DateO-getTimeO; 

long lastThreeHours - Util.MILLISECONDSJN_SECOND * 

Util.SECONDSJN_MINUTE * 
Util.MINUTES_IN_HOUR * 
3; 

. long playedTime = 0; 
Date playedAt; 

for (int i = 0; i < abunch.sizeO; i++) 
{ 

info = abunch.eIementAt(i); 

playedAt = lastPiayed.get(info.songlD); 

// don* play songs twice within 3 hours 

if (playedAt = null || (now - playedAt.getTime()) > lastThreeHours) 
A 

if (playBadWords || Hnfo.hasExplicitLyricsO) 
{ 

data = initSongGetData(info.songID, SongiUN RATED); 

if (data I- null) 
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{ 

data.setlnfo(info); 

unrated.addElement(data); 

added++; 

} 

} 

} 

} 

Util.debug("import popular added " + added + n songs"); 

} 

public void sortVectorsO 
{ 

sort(explicit, 0, explicit.size() - 1); 
sort(implicit, 0, impIicitsizeO - 1); 
sort(unrated, 0, unrated:size() - 1); 

// Util.debugO'after sorting, ratedVector is: " + ratedVector.toStringO); 

II Util.debug("after sorting, unratedVector is; " + unratedVector.toStringO); 

ordered = true; 

- : public void sort(Vector a) - -' ' : • - • - - • — 

{ 

sort(a, 0, a.sizeQ - 1); 

} 

private void sort(Vector a, int from, int to) 
{ 

//quicksort 

// If there is nothing to sort, return 
if ((a = null) || (a.size0 < 2)) return; 
int i = from, j =to; 

SongData center = (SongData) a.elementAt((from + to) / 2); 
do { 

while((i < to) && (center.score < ((SongData) a.elementAt(i)).score)) i++; 
while(G > from) && (center.score > ((SongData) a.e!ementAt(j)).score)) j-; 
if(i<j){ 

SongData temp = (SongData) a.elementAt(i); 
a,setElementAt(a.elementAt(j), i); 
a.setElementAt(temp, j); // swap elements 

} 

if(i<=j){i++;j-;} 
} whi!e(i <-j); 

if (from < j) sort(a, from, j); // recursively sort the rest 
if(i<to)sort(a, i,to); 

} 

public String toStringO 
{ 

String result = M "; 
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for (Enumeration e = hash.keysO; e.hasMoreEIements() ;) { 

int songlD = ((Integer) e.nextElementO)intVaIue(); 
Song song = get(songID); 

result = resuIt.concat("songID " + songlD 

+ " = " + song.toStringO 
+ Util.newLine); 



} 



return result; 



public String sourceCount() 
{ 

IntHash counts = new IntHashO; 
String explicitList = ""; 

for (Enumeration e = hash.keysO; e.hasMoreElementsO ;) { 

Song song = get(((Integer) e.nextEIement()).intValueO); 



if (song.getTypeQ ■= Song.EXPLICIT) . ; • ; l . 

explicitList = explicitListxoncat(song.songLD + "); 

} 

counts.increment(song.type); 

} 

return "counts: EXPLICIT = " + counts.get(Song.EXPLICIT) 
+ M (" + explicitList +") M 
+ " IMPLICIT = " + counts.get(Song.IMPLICIT) 
+ " EXCLUDED = " + counts.get(Song.EXCLUDED); 



public void toMatrix(ServletOutputStream out, int songType, int dispIayType) 
{ 

String delim = " n ; 
String prefix = MH ; 
String suffix = ""; 
String rowPrefix = 
String rowSuffix^""; 
String bold = ,,M ; 
String unbold = ""; 

if (dispIayType = Util.DISPLAY_HTML) 
{ 

delim = "</TD><TD>"; 

prefix = "<TABLE CELLPADDING= 1 CELLSPACING=0>"; 

suffix = M </TABLE>"; 

rowPrefix = "<TRXTD> M ; 

rowSufFix = M </TD></TR>"; 

bold = M <B><FONT SIZE=\M\ M >"; 

unbold = n </F(MT></B> n ; 

) 

else 

{ 
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delim = "\t"; 

} 

UtiLout(out, prefix); 

String header = Util.newLine + rowPrefix + bold 

+ Utiljoin(unbold + delim + bold, SongData. names ArrayO) 
+ unbold + rowSuffix; 



Vector v - null; 

if (songType = Song.EXPLICIT) 

v = explicit; 
else if (songType = Song.IMPLICIT) 

v - implicit; 

else 

v = unrated; 

if(v!=nu!l) 
{ 



for (int i = 0; i < v.sizeO; { 

SongData data = (SongData) v.elementAt(i); 

if(i%40 = 0) 

Util.out(out> header); 

Util.out(out, data.toDisplayString(displayType, (i + 1))); 

} 



} 

UtiI.out(out, suffix); 

} 

470 

} 

Populationjava Page 9 of 9 11/05/99 1:38 PM 
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Rating 

package com.launch.PlaylistGenerator; 
public class Rating 

{ 

protected short rating; 
protected boolean set = false; 
public RatingO 
{ 

} 

/** 

* create one with a default 
*/ 

public Rating(short defaultRating) 
{ 

rating = defaultRating; 

} 



public boolean isSetO 
. return set; 

} 

public void set(short newRating) 
{ 

rating = newRating; 
set -true; 

} 

public short getQ 
{ 

return rating; 

} 

public String toStringQ 
{ 

if(!set) 

return rating + "(Not Set)"; 

else 

return ,,M + rating; 

} 

1 

Ratingjava Page 1 of 1 1 1/05/99 1 :28 PM 
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RatingsCache 

package com. launch. PlaylistGenerator; 
import java.util.*; 

import javax.servIet.ServletOutputStream; 
import javajo.IOException; 

public final class RatingsCache implements GetRatingsCacheUsers Interface, Constants 
{ 

/** 

* This Hashtable will be of the form 

* (Integer userlD, Hashtable CachedRating objects), if the Data in 

* the cache is invalid the entry will be of the form 

* (Integer userlD, INVALIDJDATA) 

* <br> 

* The Hashtable of CachedRating objects is of the form (Integer itemID, CachedRating) 
**/ 

private Hashtable ratingsList = new Hashtable(l); 
private GetRatingsCacheUsers gtu; 

private FrequencyCounter fireq_counter = new 
FrequencyCounter(RATINGS_CACHE_INITIAL^SIZE); 

private Date lastUpdated - new DateQ; ~ ' \ " " *" 

private Date lastReset = new DateQ; 



II- 



pad with zeros. 



public RatingsCacheO 
{ 

gtu - new GetRatingsCacheUsers(this); 

//the following line is for testing purposes only, rem it out otherwise. 

gtu.SLEEP_TIME=5*60* 1 000; 

gtu.start(); 

} 

/** 

* This method will get a list of rating for the given userids 

* @param userid an array of ints representing userids, each entry should be a valid userlD, do not 

* @return a Vector of CachedRating objects 

public final Vector getRatings(Vector users) 

{ ■ 

//algorithm 

// check for userid in hashtable 

// if found add to vector of ratings 

// else build list of unfound things 

// get list of unfound things from database 

Vector allRatings = new VectorO; 

Integer userlD; 

Hashtable ratingProfile; 

Vector nonCachedUsers = new Vector(users.size()); 

Date startDate = new DateQ; 

Enumeration e - users.elementsO; 

while (e.hasMoreEIementsO) 

{ 

userlD = (Integer) e.nextEIementO; 
ratingProfile = (Hashtable) ratingsList.get(userlD); 
if (ratingProfile = null) 

{ 
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Util.debug("RatingsCache MISS on user w + userlD); 
nonCachedllsers.addElement(userlD); 

} 

else 
{ 

// benchmarkdatel = new DateO; 

Util.debug("RatingsCache HIT on user " + userlD); 

appendToVector(alIRatings, ratingProfile.elementsO); 
// Util.printEIapsedTimeO'Get from cache, " + temp_hash.size() + " 

entries", benchmark_date 1 ); 

} 

freq_counter. increment Value(userl D); 

} 

if (nonCachedUsers.sizeO > 0) 
{ 

MergeVectors(allRatings, getRatingsFromDatabase(nonCachedUsers)); 

} 

UtiI.printElapsedTime(Thread.currentThreadO-getNameO + '\ got " + allRatings.size() + 

w ratings \ startDate); 

return allRatings; 

— ■ _ 



85 setCachedUserIDs(v); 
} 

public Hashtable getMostFrequentlyUsedUsers(int i) 
{ 

Hashtable h = freq_counter.getLargest(i); 
90 Vector v =?=new Vector(h sizeO); 

// when we do this, also refresh the cache 
// to clean out any lingering data corruption 

95 Util.debug(new DateO-toStringO + " Resetting ratings cache"); 

// clear the users in the cache 
setCachedUserlDs(v); 

too lastReset - new DateO; 

// put user hash into vector 
appendToVectorCv, h.keysQ); 

ids // get all the ratings 

setCachedUserlDs(v); 

return h; 

} 

no /»• 
* 

**/ 

public final void setCachedUserIDs(Vector userlDs) 
{ 

us lastUpdated = new DateO; 

Vector cachedUsers = (Vector) userlDs.cloneO; 
Date benchmark_date = new DateO; 
if (cachedUsers.sizeO <= 0) 

120 { 

ratingsList = new Hashtable( 1 ); 
Util.debugCsetCachedUserlDs: no users passed"); 
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return; 

Enumeration e = ratingsList.keys(); 
Integer userlD; 

// find the differences between the users already in the cache 

// and the new list of users 

// leave that result in cachedUsers 

// interate through each user in the current cache 

while (e.hasMoreElementsO) 

{ 

userlD = (Integer) e.nextEIement(); 

// are they in the new list? 

if (cachedUsers. contains(user ID)) 

{ 

// cool, just remove them from the new list 
cachedUsers.removeElement(userlD); 

} 

else 
{ 

// they've been removed 

^- ^^ ■■< -< ~ — . — — - - ~> ratingsList.remove(userID); ~~™~ ~ — ™ r ^~^ — 



Vector newRatings = new VectorO; 

// get all the ratings for the new cached users 

150 

if (cachedUsers.sizeO > 0) 
{ 

newRatings - getRatingsFromDatabase(cachedUsers); 
e = newRatings.elementsO; 
155 while (e.hasMoreElementsO) 

{ 

putIntoCache((CachedRating) e.nextElementO); 

} 

} 

i6o else 

{ 

Utii.debug(new DateO-toStringO + n setCachedUserlDs: no new users in 

cache"); 

} 

165 UtiI.printElapsedTime("refreshed cached users and loaded " + newRatings.size() + " 

entries*, benchmark_date); 
} 

/** 
* 

no **/ 

private final Vector getRatingsFromDatabase(Vector userlDs) 
{ 

// 

//algorithm 

173 // 

// query database for info 
// build vector from resultsets. 

Vector results = new Vector(RATINGS_CACHE_INITIAL_S[ZE); 
Date benchmark_date = new DateO; 
i8o // — get item rating — 

GetltemRatingsFromDB itemRatings = new GetItemRatingsFromDB(userIDs f 



results); 



// — get song rating — 

GetSongRatingsFromDB songRatings = new GetSongRatingsFromDB(userIDs, 
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results); 

songRatings-Start(); 
itemRatings.start(); 
// — must wait for the two threads to finish — 
try 

{ 

itemRatings.joinO; 
songRatings.join(); 

} 

catch (InterruptedException e) 
{ 

System.err.println( M PiaylistCache: interrupted waiting for ratings, I'm 

not cleanning up..."); 

} 

// — done getting just return values — 

Util.printElapsedTime("GetRatingsFromDatabase, " + resuIts.sizeO + " entries", 

benchmark_date); 

return results; 

} 

/** 

* appends the contents of vector2 into vectorl 

private static final void Merge Vectors(Vector vectorl, Vector vector2) 

vectorl .ensureCapacity(vectorl .sizeO + vector2.size0); 
Enumeration e = vector2.elements0; 
while (chasMoreEIementsO) 
{ 

vectorl .addElement(e.nextEIementO); 

} 

} 

public static final void appendToVector(Vector v, Enumeration e) 
{ 

while (chasMoreEIementsO) 
{ 

v.addEIement(e.nextElementO); 

} 

} 

public static final String GetVectorAsCommaDeIimitedList( Vector v) 
{ 

if(v=null)returnC ,,, ); 
String s=v.toString(); 
int vector_lerigth=s.length(); 
if (vectorjength >= 3) 

{ 

return(s.substring( 1 , vectorjength- 1 )); 

} 

else 
{ 

return(""); 

} 

} 

/** 

* This method adds the value to the hashtable pointed to by the key, if the key does not exist yet it 
will create the first entry and the Hashtable 

**/. 

public final void putIntoCache(CachedRating value) 
{ 

RatingsProfile profile = null; 

Integer userlD = new lnteger(value.userID); 

// this could be more efficient if we inserted all the ratings for a particular user all at once 
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if (ratingsList.containsKey(userID)) 
{ 

profile = (RatingsProfile) ratingsListget(userlD); 

} 

else 
{ 

profile = new RatingsProfile(RATlNGS_CACHE_INITIAL_SIZE); 
ratingsList.put(userID, profile); 

} 

if (value.rating < 0) 
{ 

// unrate 

profile.remove(value.hashKeyO); 

> 

else 
{ 

profiIe.put(va!ue.hashKey0, value); 

} 

} 

public final String toStringO 
{ 

return ratingsList-toStringO;^"^-- — ~~ — - . 

public final String userListO 
{ 

String result = 

Enumeration e = ratingsListkeysO; 
Integer userlD; 

while (chasMoreElementsO) 
{ 

userlD = (Integer) e.nextEIement(); 
result = result.concat(userID + H , "); 

} 

return result; 



public final void status(ServletOutputStream out, boolean detail) throws lOException 
{ 

out.print("RatingsCache has w + ratingsList.sizeO + H users" + UtiLnewLine 

+ H Last Updated at " + 

lastUpdated.toStringO + Util jiewLine 

+ "Last Reset at " + lastReset.toStringO + 

UtiLnewLine 

+ "UserList is " + userListO + 

UtiLnewLine); 

Enumeration e = ratingsListkeysO; 
Integer userlD; 
RatingsProfile profile; 

while (chasMoreElementsO) 
{ 

userlD = (Integer) e.nextEIementO; 

out.print(Util.newLine + "Profile for userlD " + userlD + V + UtiLnewLine); 

profile = (RatingsProfile) ratingsListget(userID); 

if (profile = null) 
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{ 

310 out.printCNULL!" + UtiKnewLine); 

} 

else 
{ 

out.print(Util.newLine + profi!e.count(Constants.ITEM_TYPE_SONG) 

3i5 + " songs, M 

+ 

profi!e.count(Constants.ITEM_TYPE_ALBUM) + " albums, " 

+ 

profile.count(Constants.ITEM_TYPE_ARTlST) + ■" artists, " 

**> + profile.count((byte) 0) 

+ " total" + Util.newLine); 

if (detail) 

outprint(profiie.toStringO); 

} 

} 

} 

} 

RatingsCache.java Page 2 of 7 1 1/05/99 1:23 PM 
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RatingsProfile 

package com.launch.PlaylistGenerator; 

import java.utiI.Hashtable; 

import Java. uti I. Enumeration; 

public class RatingsProfile extends Hashtable 

{ 

public RatingsProflle(int capacity) 
{ 

supe incapacity); 

public int count(byte type) 
{ 

int count = 0; 

if(type<=0) 

return sizeO; 

else 
{ 

— — «... Enumeration e- keysO; - — — — — — « 

String key; *" 

CachedRating rating; 

while (e.hasMoreElementsO) 
{ 

key = (String) e.nextElement(); 

rating = get(key); 

if (rating.type = type) 
count++; 

} 

return count; 

> 

public CachedRating get(String key) 
{ 

return (CachedRating) super.get(key); 

} 

public String toStringO 
{ 

String result = ,,w ; 
Enumeration e = keysO; 

while (e.hasMoreElementsO) 
{ 

result = result.concat((get((String) e.nextElementO))toStringO); 

} 

return result; 

} 

RatingsProfile.java Page 2 of 2 1 1/05/99 1:35 PM 
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RatingWidgetServlet 

package com.launch.PlaylistGenerator; 

import java.util.*; 

import java.io.*; 

importjava.net.*; 

import javax.servlet.*; 

import javax.servlet.http. * ; 

/** 



* RatingWidgetServlet java 7/8/99 

* Initial Servlet for ratings Widget 

* Copyright (c) 1999 LAUNCH Media, Inc. 

* @author Jon Heiner 

* ; 

V 

public class RatingWidgetServlet extends HttpServIet implements GetRatingsCachellsersInterface, 

GetPIaylistServersInterface, Runnable 

{ 

— s ^ private Vector cachedUsers = new Vector( 1);— * ™ "»*' * ^ — ^ l .^.> wi y ^.,i l;; , :i i ; rt , ^ . o 

private GetRatingsCacheUsers gtu; \ . -r. 

private Vector playlistServers = new Vector(l); " ^'"^ 

private GetPlaylistServers gps; — 
/** This vector contains CachedRating objects */ 

private Vector dirtyRatings = new Vector(Constants.RATING_UPDATE_LIST_INITIAL_SI2E); 
private Thread myThread; 

* //. . . 

/** 

* Handle requests... 
♦/ 

public void doGet ( 

HttpServletRequest request, 
HttpServIetResponse response 
) throws ServletException, IOException 

{ 

String sE vent; 
String sRater; 
String sRatee; 
int iRateeType; 
String sRating; 
int raterlD = 0; 

// get parameters 

sEvent = request.getParameter( ,, action M ); 
// get stream for output 
ServletOutputStream out; 
response.setContentType( M text/plain"); 
response.setHeader( M Pragma", "no-cache H ); 
response.setHeader( M Cache-controP\ "no-cache"); 
response.setHeader("Expires", "0"); 

out = response.getOutputStreamO; 

try 

{ 

DBConnection conn = new DBConnectionQ; 
ifCsEventequalsCINIT")) 

{ 

sRater = request.getParameter( M rater"); 
sRatee = requesLgetParameterC'ratee"); 

iRateeType = Integer.parselnt( request.getParameter( M ratee_type w ) ); 
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+ sRater + 
+ sRatee; 



int rating = - 1 ; // not rated 
boolean implicit = false; 
String sql = ""; 
I! SONG case 

if (iRateeType = Constants. !TEM_TYPE_SONG) 
{ 

sql = "exec spJcGetSonglnfoSummary xsxx ' 



} 

else if (iRateeType = Constants.lTEM_TYPE_ALBUM) 
{ 

sql = "exec spJcGetArtistOrAlbumRating^xsxx " 



+ sRatee + ' 
+ sRater; 



} 

else 



. sql = "exec sp_lcGetArtistOrAlbumRating_xsxx ' 

+ sRatee + "," 
+ sRater; 

} 

DBResultSet rs = conn.executeSQL(sql); 
if(!rs.getBOFO&& !rs.getEOF0) 

rating = rs-getlntCVating"); 
out.printIn("rating_value= n + rating + 

"&lmplicit_indicator=not_implicit"); 

} 

else if (sEventequals("RATING_EVENT")) 
{ 

/* Do update to LaunchCast Ratings Database */ 
sRater = requestgetParameter("rater"); 



for invalid user: " + sRater); 



for invalid user: " + raterlD); 



try 
{ 

raterlD = Integer.parseInt(sRater); 

} 

catch (NumberFormatException e) 
{ 

throw new Exception("RatingWidgetServlet: rating received 

} 

if(raterID<=0) 
{ 

throw new Exception( n RatingWidgetServ!et: rating received 

> 

sRatee =» requestgetParameter("ratee"); 

iRateeType = Integer.parselnt( request.getParameter("ratee_type t, ) ); 
sRating - ^equest.getParamete^( , '^ating ,, ); 
//song case 

if (iRateeType = Constants.ITEM_TYPE_SONG) 
{ 

App. 2-148 



135 



WO 01/35667 PCT/US00/30919 

173 

conn.executeUpdate("exec sp_lcRateSongUser_isux " 

+ raterlD + V 

+ sRatee + V 

+ sRating, true); 
} 

// album case 

else if (iRateeType = Constants.ITEM_TYPE_ALBUM) 
{ 

conn.executeUpdate( w exec sp_lcRateItemUserJsux " 

+ raterlD + " t " 
+ sRatee + V 

+ sRating, true); 

} - . 
II artist case 
else 



conn.executeUpdate("exec spJcRateltemUserjsux " 

+ raterlD + "," 

+ sRatee + V 

+ sRating, true); 
} 

out.println("confimiation=rating^conilrmeci"); 
if (cachedUsers.contains(new Integer(raterID))) 

155 { 

CachedRating cr = new Cached Rating(raterID, 
Integer.parseInt(sRatee), Byte.parseByte(sRating), (byte)iRateeType); 

dirtyRatings.addElement(cr); 

Util.debug(" Added change to ratings cache update queue : " + 

i6o cr); 

} 

} 

else 
{ 

i(a otrt.println("eiTor"); 

} 

conn.closeO; 

} 

catch(DBException e) { 
170 out.println("DBException: " + e.getMessageO); 

System.err.println(new DateO-toStringO + M DBException in 
RatingWidgetServlet: " + e.toStringO); 

} 

catch(Exception e) { 
i7s out.println( H Exception raised: " + e); 

System.err.println(new DateQ-toStringO + " Exception in RatingWidgetServlet: 

M + e.toString()); 

} 

outxloseO; 

180 } 

public void init (ServIetConfig conflg) 
throws ServletException { 
super. init(conflg); 
try { 
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// 
// 



gtu = new GetRatingsCacheUsers(this); 
gps = new GetPlaylistServers(this); 

// the following 2 lines are for testing purposes only, rem them out otherwise. 
gtu.SLEEPjnME=l *20* 1000; 
gps.SLEEP_TIME=i ♦20*1000; 
gps.startO; 
gtu.startO; 

myThread = new Thread(this); 
myThread.startO; 

} 

catch (Exception e) { throw new ServletException 0; } 

} 

/** 

* Destroy method - 

* get rid of the api 

* servlets "should have" a destroy method for garbage collection 
*/ 

public void destroyO { 
gps.stop0; 
gtu.stopO; 

} 



public void updateCachedUsers( Vector topUsers) ,_„.-. 

cachedUsers = topUsers; 

} \ ' 

public void updatePlaylistServers(Vector v) 

{ 

playlistServers = v; 

} 

public void runO 
{ 

// once every N minutes go update all cached ratings with some new ratings 
Util.debug("RatingWidgetServlet notify playlistgenerators of changed rating - thread 



try 
{ 



Vector temp_dirty_ratings; 
Enumeration enum; 
Sockets; 

ByteArrayOutputStream baos; 
ObjectOutputStream oos; 
OutputStream os; 
BufTeredWriter bw; 
byteb[]; 

String serverip = null; 
while (dirtyRatings != null) 
{ 



try 
{ 

if (dirty Ratings.sizeO > 0) 
{ 

baos = new ByteArrayOutputStream(l 000); 
oos = new ObjectOutputStream(baos); 
tempdirtyratings = dirtyRatings; 
dirtyRatings =new 
Vector(Constants.RATING_UPDATE_LIST_INITIAL_SlZE); 

// need to send info to cached servers here. 

oos.writeObject(temp_dirty_ratings); 

oos.flushO; 
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b=baos.toByteArray(); 

enum = playlistServers.elementsO; 

while (enum.hasMoreElements()) 

{ 

try // this nested try / catch is so if one server 

is down the others get updated too. 

{ 

server_ip=(String)enum.nextElementO; 

Util.debug(new DateOtoStringO + 
,f Rating WidgetServlet: Sending changed ratings to : " + serverjp + n this vector : " + temp_dirty_ratings); 

s=new Socket(server_ip 3 

Constants.PORTNUMBER); 



OutputStreamWriter(os)); 

bw.write(Constants.POSTHEADER); 



bw.write(com.launch.misc.constants.USER_AGENT + " + 
com.Iaunchjniscxonstants.RATING^WIDGE 



os^s.getOutputStreamO; 
bw=new BufferedWriter(new 



bw.newLineO; 



. bw.newLineO; - 

^ " ~, - _ ~ bw.write(Xontent-length: " + 

b.Iength); 

bw.iiewLineO; 

bw.newLineO; 

bw.flushO; 

os.write(b); 

os.flushO; 

osxIoseQ; 

} 

catch (Exception el) 
{ 

System.err.printin((new 

DateO)toStringO + " Error contacting ratings cache at " + serverjp); 

//e 1 .printStackTraceO; 

} 

} 

} 

} 

catch (Exception e2) 
{ 

System.err;println((new DateO).toStringO + n Error in 

Rating WidgetServlet CacheUpdater while loop"); 

e2.printStackTrace(); 

> 

Thread.sleep(Constants.PROPAGATE_DIRTY_RATING_SLEEP_TIME); 
} 

} 

catch (Exception e) 
{ 

System.err.println(new DateO-toStringO + " Fatal Error in Rating WidgetServet 

updater thread "); 

e.printStackTraceO; 

} 

UtiLdebug("RatingWidgetServlet notify playlistgenerators of changed rating - thread 

done"); 

} 

public Hashtable getMostFrequentlyUsedUsers(int i) 
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{ 

3io return null; 

} 

} 

/♦eof*/ 
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RecList 

package com.launch.PIaylistGenerator; 

import java.uti I. Vector; 

/** 

* Launch Media, Inc Copyright 1999 

* Recommendation List - class which encapsulates 

* recommendations coming from the net perceptions engine 
* 

* RECOMMENDED USAGE 

* to access values within a RecList object: 



* void someFunction(RecList aRec) { 
* 

* if ( aRec.setToFirstRecO ) { 

* do{ 
System.out.println( aRec.getldentifierO + " : " + aRec.getPredictedRatingO ); 

* } while aRec.incrementO ; 



* ■ - 



* The "prediction result" object in net perceptions is NOT 

* persistent so is unusable outside of a carefiilly controlled 

* environment 
* 

* Further, developers within LAUNCH should not be exposed 

* to Net Perceptions data structures (as they are ugly) 
* 

* file: JaunchNetP.java 

* @author Jon Heiner 

* @since 7-30-99 
*/ 

public class RecList { 

private final static int kGrowVectorBy = 4; 
private Vector theRecs; 
private int theNumRecs = 0; 
private int thelndex = 1; 
/* Rec — inner class 

* encapsulates the ID and predicted 

* value for the item in the list; 

* the inner values are made public 

* for convenience; they are exposed 

* to this class, but are not intented 

* to be used outside of this implementation 
V 

public class Rec { 

public int thelD; 
public float the Value; 
/* Rec - creation method 

* the variables should be immutable 

*/ 

public Rec(int ilD, float fValue) { 
theValue = fValue; 
theID = i!D; 

} 

} 

/** RecList - creation method 
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* creates an empty rec list, which we will then add 

* Recs to; if you try to pull values from this it will 

* indicate that this is not possible 
*/ 

public RecListO { 

theRecs = new Vector(0, kGrowVectorBy); // create an empty vector 

} 

/** RecList - creation method w/ args 

* creates a rec list with one element; use the add 

* method to add more values to it 
♦/ 

public RecList(int ilD, float fValue) { 

theRecs - new Vector(0, kGrowVectorBy); // create an empty vector 
this.add(iID, fValue); 

} 

/** compact 

* called once the RecList has been created and 

* all items are added 
*/ 

public void compactO { 

theRecs.trimToSizeO; 

/** setToFirstRec ' . .' 

* called to set us to the first rec : ' < * - — . . ^ , - ------- _ ^ ^ v- zr. 

* if this returns false, then there are 

* no recommendations in the list 
*/ 

public boolean setToFirstRecO { 
thelndex = 0; 

if (theNumRecs > 0) return true; 
return false; 

} • 

/** increment 

* moves the internal pointer to the next item 

* returns false if there are no more'Recs in 

* the'list 
*/ 

public boolean incrementQ { 
thelndex++; 

if (thelndex < theNumRecs) return true; 
return false; 

} 

/** getidentifier 

* , returns the item ID for the current item 

* in the Rec List 

7 

public int getldentifierO { 

return (int) ((Rec) theRecs.eIementAt(theIndex)).theID; 

} 

/** getPredictedRating 

* returns the percentage value which is the 

* predicted value 
*/ 

public float getPredictedRatingO { 

return (float) ((Rec) theRecs.elementAt(theIndex)).theValue; 

} 

/** add 

* adds a new value to the Rec list 

* returns false if the values entered 

* are invalid; (e.g.: ild < 0) 
V 
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public void add(int ilD, float (Value) { 
theNumRecs++; 

theRecs.addE!ement(new Rec(iID, fValue) ); 

} 

/** length 

* returns the number of elements in the Rec list 
*/ 

public int lengthO { 

return theNumRecs; 

} 

/** createStubRecList 

* used to return "good" bogus values rather 

* than values generated from Net Perceptions 

* useful for testing and stubbing 
♦/ 

public static RecList createStubRecListO { 

RecList aRecList = new RecList(74082, (float) 0.5); 

aRecList.add(l 16377, (float) 0.6); 

aRecList.add(1233 12, (float) 0.7); 

aRecList.add(899, (float) 0.8); 

— -~™aRecList.add(58075, (float) 0.9); ■ ^~r«~~~~™~~*~ 

. return aRecList; 

/** test 

* test class 
V 

public static class Test { 
/* 

public static void main(String Q args) { 
System.out.println( "debug 0"); 
RecList aRec == createStubRecListO; 

System.out.println( "debug 1"); 
if ( aRecsetToFirstRecO ) { 
System.outprintIn( "debug 2"); 
do{ 

System.out.println( "debug 3"); 

System.out.printIn( aRec.getldentifierO + " : " + aRec.getPredictedRatingO ); 

System.out.println( "debug 4"); 

} while ( aRec.incrementO ); 

> 



} 
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SaveClips 



package com. launch. Play listGenerator; 

import java.util. Vector; 

import java.util. Date; 

public class SaveClips extends Thread 

{ 

Vector clips; 
String storedProc; 
int ordinal; 
short mediaType; 
int userlD; 

public SaveClips(Vector clips, String storedProc, int ordinal, short mediaType, int userlD) 
{ 

thisxlips = clips; 
this.storedProc = storedProc; 
this.mediaType - mediaType; 
this.userlD = userlD; 
this.ordinal = ordinal; 

public void runO 

Date startDate = new DateO; 

ThTeadxurrentThreadO.setNameC^SaveCHps for " + storedProc); 

int rowCount = 0; 

if (clips.sizeO <~ 0) 
return; 



try 
{ 



DBConnection conn = new DBCbnnectionO; 
String sql = M "; 

Clip aClip; 

for (int i = 0; i < clips.sizeO; 
{ 

aClip = (Clip) clips.elementAt(i); 

sql = sql.concat( M exec " + storedProc + " M 
+ ordinal + V 

+ aClip.media,getID(mediaType) + 1 
+ mediaType + " 
+ userID); 

ordinal-H-; 
rowCount++; 



conn.executeSQL(sqI); 
connxIoseO; 

} 

catch (DBException oops) 
{ 

Util.debugCDB Exception: " + oops.getMessage()); 
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} 

UtiI.debug(Thread.currentThreadOgetNameO + w saved M + rowCount + M clips"); 
Util.printElapsedTime(Thread.currentThreadO getNameO, startDate); 

} 

} 

SaveClipsjava Page 2 of 2 1 1/05/99 1:25 PM 
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SavePIaylist 

package com. launch. Play I istGenerator; 

import java.util.Date; 

public class SavePIaylist extends Thread 

{ 

Playlist list; 

int ordinal, to, from; 

public SavePlaylist(PIaylist list, int from, int to, int ordinal) 
{ 

this.list = list; 
this.ordinal = ordinal; 
this.to = to; 
this.from = from; 

} 

public void run() 
{ 

Date startDate = new DateQ; 

Tru-eadxurrentThread0.setName( H SaveP!aylist ( H + from + "- n + to + ")"); 



int rowCount ~ 0; 



try 
{ 

DBConnection conn = new DBConnectionO; 
String sql = MM ; 

SongData data; 
short origin; 

for (int i = from; i < to; i++) 
{ 

data = (SongData) listmedia.elementAt(i); 

if (listpopularOnly) 

origin - (short) SongData. SOU RCE_FORCED_POPULAR; 

else 

origin = (short) data.origin(); 

if (data.querySource = SongData.SOURCE_RATED) 
origin = (short) data.rating.getSourceO; 

// 

sql = sqlxoncatC exec sp_lcSaveMediaPlaylist_ixxd " 
+ ordinal + tt , " 

+ data.getMediaID(iistmediaType) + M , " 
+ list.mediaType + H , " 
-HistuserID+-\ " 
+ data.impIicit+ , ', M 
+ origin); 

ordinal++; 
rowCount++; 

} 

conn.executeSQL(sql); 
conn.closeO; 

} 

catch (DBException oops) 
{ 

UtiI.debug( M DB Exception: " + oops.getMessage()); 
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} 

Util.debug(Thread.currentThread().getNameO + " saved " + rowCount + " songs"); 
Util.printElapsedTime{Thread.currentThread().getNameO, startDate); 

} 

} 

SavePlaylistjava Page 2 of 2 1 1/05/99 1 :25 PM 
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SimpleClip 

package com. launch. Play listGenerator; 

import java.io.Serializable; 

public class SimpleClip implements Serializable 

{ 

int medialD; 
int ID; 
byte origin; 

public String toStringO 
{ 

return "clipID- # + ID + mediaID= M + medialD + origin= H + origin; 

} 

/** 

* Contructor for ads, news, tips 
*/ 

public SimpleClip(int ID, int medialD) 
{ 

this. medialD = medialD; 

this.ID = ID; — -< — . ■ '« — — — 

} . . • • . . 

/** 

* Constructor for songs 
*/ 

public SimpIeClip(int ID, int medialD, byte origin) 
{ 

this(ID, medialD); 
this.origin = origin; 

} 

} 

SimpleCIip.java Page I of 1 11/05/99 1:32 PM 
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SimpleClipList 

package com. launch. Play listGenerator; 

import java.util.Vector; 

public class SimpleClipList extends Vector 

{ 

public Simp!eClipList(int size) 
{ 

super(size); 

} 

public SimpIeClip popO 
{ 

if(sizeO>0) 
{ 

SimpIeClip clip = (SimpIeClip) e!ementAt(0); 

rernoveElementAt(0); 

return clip; 

} 



return null; 

} 

SimpIeClipListjava Page 1 of I 1 1/05/99 1 :32 PM 
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SimplePlaylist 

package com. launch. Play listGenerator; 
import java.util.Vector; 
import java.io.Serializable; 
import java.io.ByteArrayOiitputStream; 
import java.io.ObjectOutputStream; 
import java.io.ObjectlnputStream; 
import java.io.ByteArraylnputStream; 
import java.utiI.Date; 

public class SimplePlaylist implements Serializable 
{ 

SimpleCIipList news = new SimpIeClipList(lO); 
SimpleClipList ads = new SimpleClipList(IO); 
SimpleCIipList tips = new SimpleClipList(l 0); 
SimpleClipList songs = new SimpleClipList(50); 

Date lastAd; 
Date lastNews; 
DatelastTip; 



short mediaType; 

intmoodID; 1 " ' ' " 

int djID; 

public String toStringO 
{ 

return M ads= H + ads.toStringO + "» M + 

"news=" + news.toStringO + n , " + 
M songs= H + songs.toStringO + " + 
M tips=" + tips.toStringO; 

} 

public void resetDates(Date newDate) 

{ 

lastAd = lastNews = lastTip = newDate; 

} 

public void save(int userlD) 

{ .. 

ay 
{ 

DBConnection conn = new DBConnectionO; 
save(conn, userlD); 

> 

catch (DB Exception e) 
{ 

System.eiT.println(new DateO-toStringO + " DBException in SimplePlaylist:save: " 

e.toStringO); 

e.printStackTraceO; 

} 

} 

public void save(DBConnection conn, int user ID) 
{ 

try 
{ 

String sql = "exec sp_lcSavePlaylist_lxxd " + userlD + ?"; 
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DBPreparedStatement statement = conn.prepareStatement(sql); 

byte[] b = toByte ArrayO; 

statement. setBytes(l, toByteArrayO); 

statement. executeUpdateO; 

} 

catch (DBException e) 
{ 

System.err.println(new DateQ-toStringO + " DBException in SimplePlaylistrsave:" 



e.toStringO); 
} 



} 



public static SimplePlaylist fromBytes(byte[] b) 
{ 

if (b = null j| b.length <= 0) 
return null; 

T 



ByteArraylnputStream bais new ByteArraylnputStream(b); ~ ' * 

if (bais = null) 

return null; 

ObjectlnputStream ois = new ObjectlnputStream(bais); 

if(ois = null) 

return null; 

return (SimplePlaylist) ois.readObjectO; 

} 

catch (Throwable e) 
{ 

System.err.printlnC'Exception in SimplePlay!ist:fromBytes: M + e.toStringO); 

} 

return null; 

} 

public static SimplePlaylist Ioad(DBConnection conn, int userlD) 
{ 

String sql = "exec spJcGetPlaylist_xsxx '* + userlD; 

try 
{ 

DBResultSet rs - conn.executeSQL(sql); 

return SimplePlaylist.fromBytes(rs.getBytes("playlist")); 

} 

catch (Throwable e) 
{ 

System.err.println("Exception in SimplePlaylist:load:" + e.toStringO); 

} 

return null; 

} 

private bytef] toByteArrayO 
{ 

try 
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{ 

ByteArrayOutputStream baos = new ByteAirayOutputStreamO; 
125 ObjectOutputStream oos = new ObjectOutputStream(baos); 

oos.writeObject(this); 
return baos.toByteArrayO; 

} 

catch (Throwable t) 

130 { 

System.err.println("toByteArray died: " + t.toStringO); 

t.printStackTraceO; 
return null; 

} 

SimplePlaylistjava Page 3 of 3 1 1/05/99 1 :35 PM 
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Song 

package com.Iaunch.PlaylistGenerator; 

public class Song 

{ 

public final static short EXCLUDED = 4; 
public final static short EXPLICIT = 3; 
public final static short IMPLICIT = 2; 
public final static short UNRATED = 1; 
public final static short ANY =0; 

public int songID; 
public short type = ANY; 
private SongData data = null; 

public Song(int songID, short type) 
{ 

this.songID = songID; 
setType(type); 

} 



public String toStringO 

return "Song " + songID 
+ ",rype = " 
+ typeStringO 
+ \data = " 

+ ((data — null) ? "null" : data.toStringO); 

} 

public String typeStringO 
{ 

switch (type) 
{ 

case ANY: 

return "ANY"; 
case EXPLICIT: 

return "EXPLICIT"; 
case IMPLICIT: 

return "IMPLICIT"; 
case UNRATED: 

return "UNRATED"; 
case EXCLUDED: 

return "EXCLUDED"; 

default: 

return "UNKNOWN"; 

} 

} 



// this should wait for setType 
public SongData getDataO 

{ 

return data; 

} 

// this should wait for setType 

public short getTypeO 

{ 

return type; 

} 
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// returns whether or not this is suitable for setting SongData 

public boolean setType(short newType) 

{ 

short oldType = type; 

if (newType = type) 

return true; 
else if (newType < type) 

return false; 

else 

type = newType; 

// add or delete song data 

if (newType — EXCLUDED) 
{ 

// if(oldType!=0) 

// Util.debug(Thread.currentThreadO-getNameO + ": deleting data for song M h 

songID + ", oldType was " + oldType); 

data = null; 

} 

. else if (oldType = ANY && (newType = EXPLICIT || newType = IMPLICIT || newType 
UNRATED)) 

{ ' ~ " — - ; 

data = new SongData(songID); 

} 

return true; 

} 

} 

Song.java Page 2 of 2 1 1/05/99 1:26 PM 



App. 2-1*6 



WO 01/35667 



191 



PCT/US00/30919 



SongData 

package com.launch.PIaylistGenerator; 

public class SongData 

{ 

intsongID; 

byte querySource; 

public AverageRating djsAverage; 

double score, 

netp, 

implicit, 

confidence, 

IastPlayed, 

bds, 

ratingF, 

djsF, 

netpF, 

commRatingF, 

IastPlayedF, 

bdsF; 

— - private Songinfo info; * • r ■■. ' , „„ ■ .,, „ , — „ .^ . .r ^ 

private Rating djs = new Rating((short) Constants.DEFAULT_DJS_SCORE); 

private byte djSource = SOURCEDJS; — ■ - — ^— 

public SongRating rating; 
PickStatus status; 

public final static byte SOURCE_RATED =1; 
public final static byte SOURCEJMPLICIT_ALBUM = 2; 
public final static byte SOURCEJMPLICIT_ARTIST = 3; 
public final static byte SOURCE_IMPLICIT_SONG = 4; 

public final static byte SOURCE_DJS =5; ■ fc 

public final static byte SOURCE_DJS_SONG =5; 

public final static byte SOURCE_BDS =6; 

public final static byte SOURCE_POPULAR = 7; 

public final static byte SOURCE_RANDOM =8; 

public final static byte SOURCE_NETP = 9; 

public final static byte SOURCE_ALL = 10; 

public final static byte SOURCE_RECENTLY_PLA YED =11; 

public final static byte SOURCE_FORCED_POPULAR =12; 

public final static byte SOURCE_GENRES = 13; 

public final static byte SOURCE_DJS_ALBUM = 14; 

public final static byte SOURCE_DJS_ARHST « 15; 

public final static byte DO_NOTHING =0; 

public final static byte MAKE__ME_IMPLICIT = 1; 

public final static byte EXCLUDE_ME = 2; 

public SongData(int songlD) 

{ 

IastPlayed = Constants.DEFAULT_LASTPLAYED_SCORE; 

djsAverage = new AverageRating((short) Cpnstants.DEFAULT_DJS_SCORE); 

status = new PickStatusQ; 

netp = Constants.DEFAULT_NETP SCORE; 

this.songID = songlD; 
rating - new SongRatingO; 

} 

public boolean equals(SongData otherData) 
{ 

return (songlD = otherData.songID); 

} 

public byte originQ 
{ 

double max Value = 0; 

byte maxSource = SOURCE_RANDOM; 



App. 2-167 



WO 01/35667 PCT/US00/30919 

192 

byte ratingSource = 0; 
if (ratingjsSetO) 

ratingSource = rating.getSourceO; 

if (infoxommRating > max Value && info.commRating > Constants.POPULARJTHRESHOLD 
&& ratingSource != 1 ) 

{ 

maxValue = info.commRating; 
maxSource = SOURCE_POPULAR; 

} 

if (djs.isSetO && djs.get() >= maxValue && djs.get() > 0 && ratingSource != 1) 
{ 

maxValue =djs.get(); 
maxSource = djSource; 

} 

/* 

if (netP > maxValue) 
{ 

maxValue = netP; 

maxSource = SOURCENETP; 

} 

*/ 

^ , — ,~~~if (bds > 0 && bds >= maxValue && ratingSource != 1) — ~ - ~ 

: { . • * . 

■~ " maxValue -bds; 

maxSource = SOURCE_BDS; 

} 

// according to the weight matrix, if there's an explicit rating, 
//that's the only source 

// but let's lie to people because they dont like it when we say 

// we played lowly-rated songs for them 

// even though that's what we say we will play anyway 

if(rating.isSet()) 

{ 

short value = rating.getO; 

if (value > Constants.MIN_RATING_FOR_RATED_SOURCE && value >= maxValue) 

maxValue = value; 
maxSource = ratingSource; 

} 

} 

// lies, lies, lies. 

if (maxValue < Constants.MIN_RATING_FOR_RATED_SOURCE) 
{ 

maxSource - SOURCERANDOM; 

} 

return maxSource; 

} 

public void calcu!ateDJs(ItemsProfiIe items, AlbumArtistData albumAndArtist) 

// put in the default 
djs.set(djsAverage.getO); 
djSource = SOURCE_DJS_SONG; 
if (djsAverage.countO <= 0) 
{ 

djSource = SOURCERANDOM; 

Item albumltem = aIbumAndArtist.getAlbum(items, this); 

Item artistltem = aIbumAndArtist.getArtist(items, this); 

// dont calculate implicit ratings based on various artists 

if (artistltem != null && Artistlnfo.isVariousArtists(artistltem.itemlD)) 

{ 

artistltem = null; 

} 
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if (albumltem != null && albumltem.djsAverage.countO > 0) 

djs.set(aIbumItem.djsAverage.get()); 
djSource « SOURCE_DJS_ALBUM; 

} 

else if (artistltem !- null && artistltem.djsAverage.count() > 0) 
{ 

djs.set(artistltem.djsAverage.getO); 
djSource = SOURCE_DJS_ARTIST; 

} 

} 

} 

public byte ca!cu!ateImpIicit(ItemsProfile items, AlbumArtistData albumAndArtist) 

if(!rating.isSetO) 

Item albumltem = aIbumAndArtist.getAlbum(items, this); 

Item artistltem = albumAndArtist.getArtist(items, this); 

// don't calculate implicit ratings based on various artists 

if (artistltem != null && Artistlnfo. is VariousArtists(artist Item. itemID)) 

{ 

} 

if (albumltem != null && albumltem.userRating isSetO) 

short albumRating = albumltem.userRating.getO; 
if (albumRating = 0) 

return EXCLUDE_ME; 

else 
{ 

rating.set(albumRating, 
SongRating.RATING_SOURCE_FROM_ALBUM); 

return MAKE__MEJMPLICIT; 

} 

> 

else if (artistltem != null && artistltem.userRating.isSetO) 
{ 

short artistRating = artistItem.userRating.getO; 
if (artistRating = 0) 

return EXCLUDE_ME; 

else 
{ 

rating.set(artistRating, 
SongRating.RATING_SOU RCE_FROM_ARTI ST); 

return MAKEJ4EJMPLICIT; 

} 

} 

else if (artistltem != null && artistItem.songAverage.count() > 0) 
{ 

rating.set((short)artistItem.songAverageScore(info.aIbum.artist), 
SongRating.RATING_SOURCE_AVERAGE_SONG_RATING_BY_ARTIST); 

return MAKE_ME_IMPLICIT; 

} 

} 

return DO-NOTHING; 

} 

public void setBDS(short score) 
{ 

bds = score; 

} 

public double getBDSO 
{ 
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return bds; 

} 

public void score(WeightMatrix w, StationList stations) 
{ 

// score bds 

bds = info.bdsScore(stations); 
byte s - rating.getSource(); 
/* 

// we're not using confidence right now. Take it out for speed 
confidence = 0; 

if (ratingSource != SongRating.RATING_SOURCE_EXPLlCIT) 
{ 

if (djs N DEFAULT_DJS_SCORE) 

confidences 10; 
if(netp>0) 

confidences 10; 
if (info.commRating > 0) 

confidences 10; 

} 

*/ 

// implicit rating is based on ratings data 

— ratingF™= (rating.get0 ^w.matrix[s][WeightMatrix^ 

djsF =(djs.get0 * w.matrix[s][WeightMatrix.DJS ]); 

netpF =(netp ~ * w.matrix[s][WeightMatrix.NETP ^^.^ 

commRatingF ~ (info.commRating * w.matrixts][WeightMatrix.GOMM__RATlNG]); 

lastPIayedF » (lastPIayed * w.matrix[s][WeigntMatrix.LAST_PLAYED]); 

bdsF = (bds * w.matrix[s][WeightMatrix.BDS ]); 

implicit « ratingF + djsF + netpF + commRatingF; 

// score is based on other factors 

score = implicit + lastPIayedF + bdsF; 

confidence += w.matrix[s][WeightMatrix,CONFIDENCE]; 

} 

public void setInfo(SongInfo stuff) 
{ 

info - stuff; 

} 

public Songlnfo getlnfpO 
{ 

return info; 

} 

public boolean isInfoSetO 
{ 

return (info != null); 

} 

public int getArtistlDO 
{ 

return info.album.artist.ID; 

} 

public int getAlbumlDO 
{ 

return info.album.ID; 

} 

public String getArtistNameO 
{ 

return info.album.artist.title; 

} 

public String getAlbumNameO 
{ 

return info.album.title; 

} 

public int getMediaID(short mediaType) 
{ 
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return info.media.getID(mediaType); 

} 

public String getSongName{) 
{ 

return info.title; 

} 

public String sourceString(byte source) 
{ 

switch (source) { 

case SOURCEJRECENTLYPLAYED: 

return "recent"; 
case SOURCE_RATED: 

return "rated"; 
case SOURCE_IMPLICIT_ALBUM: 

return "album"; 
case SOURCE_IMPLICIT_ARTIST: 

return "artist"; 
case SOURCEJMPLICIT_SONG: 

return "s avg"; 
case SOURCEJXTS: 

return "djs"; 

case SOURCE DJS AL BUM: _ ^ 

return 'MjAlb"; 

^ :-v -^.--case SOURCE_DJS_ARTIST: v - ~~ ' - - - . - 

return "djArt"; 
case SOURCE_BDS: 

return "bds"; 
case SOURCEJ>OPULAR: 

return "pop"; 
case SOURCE_RANDOM: 

return "random"; 
case SOURCE_NETP: 

return "netp"; 
case SOURCE_GENRES: 

return "genres"; 
case SOURCE_ALL: 

return "all"; 

default: 

return"?"; 

} 

} 

public static String origmText(byte origin, String singuIarDJ, String posessiveDJ) 

switch (origin) 
{ 

case SOURCE_RATED: 

return (singuIarDJ + " rated this song"); 
case SOURCE JMPLIC1T_ALBUM: 

return (singuIarDJ + " rated this album"); 
case SOURCE JMPLICIT_ARTIST: 

return (singuIarDJ + " rated this artist"); 
case SOURCE JMPLIC1T_S0NG: 

return (singuIarDJ + H rated other songs by this artist"); 
case SOURCE_DJS: 

return (posessiveDJ + " DJs rated this song"); 
case SOURCE_DJS_ALBUM: 

return (posessiveDJ + " DJs rated this album"); 
case SOURCEJDJS_ARTIST: 

return (posessiveDJ + " DJs rated this artist"); 
case SOURCE_BDS: 

return (posessiveDJ + " radio stations play this song"); 
case SOURCEJ>OPULAR: 
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return "This song is popular on LAUNCHcast stations"; 
case SOURCE_RANDOM: 

return "This song is a random pick"; 
case SOURCE_NETP: 

return "Song recommendations"; 
case SOURCE_FORCED_POPULAR: 

return "Popular - choose more genres for your music"; 

} 

return ""; 

} 

public String toStringO 
{ 

return "songID:" + songID + " 

+ "score:" + score + " 
+ "implicit:" + implicit + ", " 
+ "confidence: M + confidence + ", " 

+ "lastPlayed:" + lastPlayed + *\ " 

+ "rating:" + rating + ", " 

+ "ratingSource:" + rating.getSource() + " t " 

+ H bds: w + bds + V 
+ w djs:" + djs.get() + V 

^^^^ r*^. ~ + "source:" + sourceString(querySource) + UtilnewLme;^**"*""" 

J . .' ■ , „._.., ■ ' ; ,, 

public PlaylistEntry toPlaylistEntry(short mediaType) 
{ 

PlaylistEntry result = new PlaylistEntryO; 

result.albumID - getAlbumlDO; 

resuIt.artistID = getArtistlDO; 

result.albumTitle = info.album.title; 

resultartisiTitle = info.album.artisttitle; 

result.fi lepath = info.media.getFilepath(mediaType); 

resuItmedialD = getMediaID(mediaType); 

resuit.songID = songID; 

result.songTitle = info.title; 

resulttitle = info.title; 

return result; 

> 

public SimpleClip toSimpleCiip(short mediaType) 
{ 

return new SimpleClip(songID, getMedialD(mediaType), originO); 

} 

public String toDisplayStririg(int dispIayType, int count) 
{ 

String delim =" n ; 
String prefix = Mtt ; 
String suffix = ""; 
String bgcolor = ""; 

if (dispIayType = Util.DISPLAY_HTML) 
{ 

if (count %2 = 0) 

bgcolor = "#CCCCFF"; 

else 

bgcolor = "white"; 

delim = "<7FONTX/TD><TD BGCOLOR= n + bgcolor + "xFONT SlZE=\"-2\">' 
prefix = "<TR><TD BGCOLOR-" + bgcolor + "><FONT SIZE=\"-2\">"; 
suffix = w </FONT><VTD></TR>"; 



} 

else { 
} 



delim = "\t"; 
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return (prefix + count 



375 



rating.getSourceO + ")" 



390 



395 



400 



+ del 
+ del 
+ del 
+ del 
+ deli 
deli 
+ deli 
+ del 
+ del 
+ del 

+ del 
+ del 
+ deli 
+ deli 
+ del 
+ deli 
+ deli 
+ deli 
+ del 
~+ suffix 



m + songID 

m + sourceString(querySource) 
m + sourceString(origin()) 
m + status.toDisplayString(displayType) 
m + status.order 
m + UtiI.fix(score, 2, 0) 

m + Math.round(lastPIayed) + 7" + Math.round(lastPlayedF) 
m + Math.round(bds) + 7 M + Math.round(bdsF) 
m + Math.round(implicit) 

m + Util.fix(rating.get(), 0, 2) + 7" + Util.fixfratingF, 0, 2) + " (" + 

m + Math.round(djs.getO) + 7" + Math.round(djsF) 
m + Math.round(netp) + 7" + Math.round(netpF) 
m + Math.round(info.commRating) + 7" + Math.round(commRatingF) 
m + getAlbumID0 
m + getArtistlDO 
m + getArtistNameO 
m + getSongName() 
m .+ getAIbumNameO 
m + info.album.genresStringO 



); 



415 



) 

public String originTclListO 
{ 

return. "{" + songID + " " + originO + " " + Math.round(implicit) + "} 

} 

public static StringQ namesArrayO 
{ 

Stringf] names = { "#", 

"songID", 
"query", 
"origin", 
"status", 
"ord", 
"score", 
"IastP/7 
"bds", 
"impl.", 
"rating(t) w , 
"djs", 
"netP.", 
"comm", 
"albumID", 
"artisID", 
"artist", 
"title", 
"album", 



} 



}; 

return names; 
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SongGroup 

package com. launch. Play iistGenerator; 

import java.util. Vector; 

public class SongGroup extends Vector 

{ 

public SongData pickRandom(int factor) 
{ 

int leftlnList = sizeO; 
if(leftInList<=0) 
return null; 

double rand = Utii.random(leftInList - 1) + 0.00001 ; 

int picklndex = (int) Math.round((Math.pow(rand, factor) / Math.pow(leftInList - 1 , factor)) 

* (leftlnList - I)); 

SongData pick - (SongData) elementAt(pick!ndex); 
double pickDouble = picklndex; 

pick.status.percentile = (short) Math.round((pickDouble / sizeO) * 100); 

removeElementAt(pickIndex); 

return pick; 

SongGroup Java Page 1 of 1 1 1/05/99 1:28 PM 
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Songlnfo 

package com.launch.PlaylistGenerator; 
import java.util. Vector; 
public class Songlnfo 
{ 

int songlD; 

byte commRating = Constants. DEFAULT_COMMRATfNG; 
private boolean explicit = false; 

Albumlnfo album; 
String title; 

private Vector bdsRanks; 
public MediaList media; 

public SongInfo(int songlD) 
{ 

this.songID = songlD; 
media = new MediaListQ; 

} 



^-^j- w public void addBDSRank(BDSRank rank) 

if (bdsRanks = null) 

bdsRanks = new Vector(l, 1); 

bdsRanks.addElement(rank); 

} 

public int getArtistlDQ /* throws Exception */ 
{ 

return album.artist.ID; 
/* 

if (album = null) 
{ 

throw new Exception("album is not set for Songlnfo songlD " + songlD + "( H + title + 

T); 

} 

return album.getArtistlDO; 
*/ 

} 

public int getAlbumlDO /* throws Exception */ 
{ 

/* 

if (album = null) 
{ 

throw new Exception("album is not set for Songlnfo songlD " + songlD + "(" + title + 

")"); 

} 

♦/ 

return album.ID; 

} 

public double bdsScore(StationList stations) 

{ 
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if (bdsRanks = null || stations.sizeO <= 0) , 

return Constants. DEFAULT_BDS_SCORE; 

inti =0; 

int pointBar = Constants. BDS_SCOR£_POINTBAR; 
float maxPoints = Constants. BDS_SCORE_MAX_POINTS; 
float totalpoints = 0; 
float numStations =0; 

BDSRank rank; 
Station sta; 

for (int j = 0; j < bdsRanks.size0; j++) 
{ 

rank = (BDSRank) bdsRanks.elementAt(j); 
sta = stations.get(rank. station ID); 

if (sta != null) 
{ 

totalpoints += (maxPoints - rank.rank); 
numStations++; 

\ ■ } / _ _ \. 

double potentialStations = stations.sizeO; 
double score = ((((totalpoints / potentialStations) / maxPoints) + (numStations / potentialStations) 

150.0); 

return score; 

} 

public String bdsStringO 
{ 

String result =""; 

if (bdsRanks = null) 

return "(none)"; 

for (int i = 0; i < bdsRanks.sizeQ; i++) 
{ 

result = result.concat(bdsRanks.elementAt(i).toStringO + V); 

} 

return H (" + result + ")"; 

} 

public String toStringO 
{ 

return "songID=" + songID + " 
+ "title=" + title + M 
+ "commRating= n + commRating + 
+ M media=" + media.toStringO 
+ *'bdsRanks=" + bdsStringO 
+ "album= n + album.toStringO; 

} 

public void setExplicitLyrics(booIean badStuf!) 
{ 

explicit = badStuff; 

} 

public boolean hasExplicitLyricsQ 

App. 2-176 



WO 01/35667 PCT/US00/30919 

201 

{ 

return explicit; 

} 

} 

Songlnfo.java Page 3 of 3 1 1/05/99 1:35 PM 
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SonglnfoCache 

package com. launch. Play I istGenerator; 

import java.uti I. Hashtable; 

import java.utiL Enumeration; 

import javax.servletServletOutputStream ; 

impojtjava.util.Date; 

import java.util. Vector; 

public class SonglnfoCache 

{ 

private Hashtable songs; 
private Hashtable albums; 
private Hashtable artists; 
private Songlnfo songList[]; 
private Hashtable ads; 
private Hashtable news; 
private Hashtable tips; 
private Clip adList[j; 
private Clip newsList[]; 
private Clip tipListn; 

— """-"'private IntHash mediaTypes; ~~ — — — — - - — ~~ — 

^ _ public PopularSongs popular; 

^ public RatingsCache ratingsCache; 

private Genrelndex genres; 

public final static byte TYPE_SONG = 1 ; 

public final static byte TYPE_ALBUM =2; 

public final static byte TYPE_ARTIST = 3; 

public final static byte TYPE_AD =4; 

public final static byte TYPE_NEWS =5; 

public final static byte TYPEjriP =6; 

private ServletOutputS tream out; 

public Date lastUpdate; 

public SongInfoCache(ServIetOutputStream out) 
{ 

// use memory most efficiently with load factor 1 

songs -newHashtable(50000); 

albums » new Hashtable(3000); 

artists = new Hashtable( 1500); 

ads = new HashtabieO; 

news = new HashtabieO; 

tips = new HashtabieO; 

mediaTypes = new IntHashO; 

genres =newGenreIndex(i00, 1); 

populateO; 

lastUpdate = new DateO; 
public SongList getPopu!ar(short medtaType) 

return popular.get(mediaType); 
public SongList getlnGenres(GenreList myGenres) 

return genres.getlnGenreList(myGenres); 
public SongList getInGenre(int genrelD) 

return genres.getInGenre(genreID); 
public int countInGenres(GenreList myGenres) 



return genres.countlnGenreList(myGenres); 
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} 

private void popuIateO 
{ 

try 
{ 

DBConnection conn = new DBConnection(); 

DBResultSet rs = conn.executeSQL("exec spJcoGetSongDataCache_xsxx M ); 

int songID, rnediaType, rank, stationID, rowCount; 

short genrelD; 

String filePath; 

Songlnfo aSong; 

Artistlnfo anArtist; 

Albumlnfo anAlbum; 

rowCount = 0; 

while (Irs.getBOFO && !rs.getEOF0) 
{ 

songID = rs.getInt("songID w ); 
rnediaType = rs.getInt( M mediaType"); 

aSong = (Songlnfo) init(songID, SongtnfoCache.TYPE_SONG); 
filePath « rs.getString("server w ) + rs.getString( M directory tt ) + "\\" + 

rs.getString( H filePath M ); 

„ , ^"™**™"aSong.iriedia^add((short) rnediaType, rs.getint^medialD'*), filePath); 
„ ,.v _ _ _ aSong.title ~ rs.getString( M song"); 

anArtist = (Artistlnfo) init(rs.geant( ,, artistID"), 

SonglnfoCache.TYPEJVRTIST); 

anArtist.title = rs.getString( M artistT); 
anArtist.songs.put(new Integer(songID), aSong); 
anAlbum = (Albumlnfo) mit(rs.getInt("albumID"), 

SongInfoCache.TYPE_ALBUM); 

anAIbum.title = rs.getString("aibum"); 

aSong.setExplicitLyrics(rs.getInt("expIicit")= I); 

// add year and date added 

an Album, artist = anArtist; 

aSong.album = anAlbum; 

mediaTypes.increment(mediaType); 

rowCount++; 

rs.next(); 

} 

UtiI.debug("SongInfoCache:populate loaded " + rowCount + " media"); 
rs = conn.executeSQL("exec sp_lcoGetCommRatingCache_xsxx"); 
rowCount = 0; 

while (Irs.getBOFO && !rs.getEOF0) 
{ 

songID = rs.getInt( n songlD"); 

aSong = (Songlnfo) get(songID, SongInfoCache.TYPE_SONG); 

if (aSong != null) 
{ 

aSongxommRating = (byte) rs.getlnt^commRating"); 
rowCount++; 

} 

rs.nextO; 

} 

UtiI.debug("SongInfoCache:popuIate loaded n + rowCount + " commRatings"); 
rs = conn.executeSQL("exec spJcoGetGenreCache_xsxx"); 
while (!rs.getBOF0 && Irs.getEOFO) 
{ 

genrelD « (short) rs.getlntO'genrelD"); 
songID - rs.getInt( n songID n ); 

aSong = (Songlnfo) get(songID, SonglnfoCache.TYPEJSONG); 

if (aSong != null && aSong.album != null) 

{ 



App. 2-179 



WO 01/35667 PCT/USOO/30919 

204 

aSong.album.addGenre(genreID); 
genres.add(genreID, aSong); 
rowCount-H-; 

} 

rs.nextO; 

} 

Util.debug("SongfnfoCache:popu1ate loaded " + rowCount + " genre mappings"); 
rowCount = 0; 

rs = conn.executeSQL( n exec sp_lcoGet£DSCache_xsxx"); 

while (!rs.getBOF() && !rs.getEOF0) 

{ 

songID = rs.getInt("songID M ); 

aSong = (Songlnfo) get(songID, TYPESONG); 

if (aSong != null) 

{ 

rank = rs.getInt("rank M ); 
stationID = rs.getlntfstationlD"); 
rowCount++; 

aSong.addBDSRank(new BDSRank((short) stationID, (byte) rank)); 

} 

rs.nextO; 

^ - , _ . Util.debug("SongInfoCache:popuIate loaded 

//import ads 
rowCount = 0; 

rs = conn.executeSQL("exec sp_IcoGetAdCache_xsxx M ); 
Clip ad; 
int clipID; 

while (Irs.getBOFO && Irs.getEOFO) 
{ 

clipID « rs-getlntC'clipID"); 
// filePath ~ rs.getStringO'server") + rs.getStringrdirectory") + H /" + 

^s.getString( t, filePath ,, ); 

ad =(Clip)init(clipID,TYPE_AD); 
// ad.name = ^s.getString( ,t clipName ,, ); 

ad.media.add((short) rs.getlntC'mediaType"), rs.getlnt(" medial D"), null); 

rowCount++; 

rs.nextO; 

) . 
Util.debug("SongInfoCache:populate loaded H + rowCount + " ad media"); 

// import news 

rs - conn.executeSQL^exec sp_!coGetNewsCache_xsxx M ); 
rowCount = 0; 
Clip newsbit; 

while (Irs.getBOFO && Irs.getEOFO) 
{ 

clipID = rs.getlnt("clipID"); 

filePath = rs.getStringC'server") + rs.getStringC'directory ") + "\V + 

rs.getStringCTilePath"); 

newsbit = (Clip) init(clipID, TYPE_NEWS); 
newsbit.name = rs.getString("clipName ,, ); 

newsbit.media.add((short) rs.getInt("mediaType"), rs.getlntCmedialD*'), 

filePath); 

rowCount-H-; 
rs.nextO; 

} 

UtiLdebug("SongInfoCache:popuIate loaded " + rowCount + " news media"); 
// import tips 

rs = conn.executeSQLO'exec sp_lcoGetTipCache_xsxx"); 
rowCount = 0; 
Clip tip; 
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while (!rs.getBOF0 && !rs.getEOF0) 
{ 

clipID = rs.getInt("clipID"); 

filePath = rs.getString( w server") + ^s.getString("directory ,, ) + M \\" + 

.getString( n filePath"); 

tip = (Clip) init(clipID, TYPEJTIP); 
tip.name = rs.getStringC'clipName 1 '); 

tip.media.add((short) rs.getInt( M mediaType"), rs.getInt("medialD"), filePath); 

rowCount-H-; 

rs.nextO; 

} 

UtiI.debug("SongInfoCache:popu1ate loaded H + rowCount + w tip media"); 
conn.closeO; 

} 

catch (DBException oops) 
{ 

System.outprintIn( n DBException in cache populate: " + oops.getMessageO); 

} 

. // populate the songs array 

songList = new SongInfo[songs.sizeOJ; 
inti = 0; 

fo?(Eriwrierati^^^ ~ 
~ \; songListp] = (Songlnfo) songs.get((Integer) e.nextElementO); 

} 

// populate the ads array 
adList = new CHp[ads.sizeO]; 
i-0; 

for (Enumeration e = ads.keysQ; ehasMoreElementsO ;) { 

adList[i] = (Clip) ads.get((Integer) e.nextEIementO); 
i++; 

'. > ■ 

// populate the news array 

newsList = new Clipfnews.sizeO]; 

i-0; 

for (Enumeration e = news.keysO; e.hasMoreEIementsO 0 { 

newsListfi] = (Clip) news.get((lnteger) e.nextElementO); 

} 

// populate the tips array 
tipList = new Clip[tips.size()]; 
i = 0; 

for (Enumeration e = tips.keysO; e.hasMoreEIementsO ;) { 

tipList[i] = (Clip) tips.get((lnteger) e.nextElementO); 

} * 
// make popular lists 

popular = new PopularSongs(songs, mediaTypes); 
Util.debug("SongInfoCache:popuIate done"); 

} 

private Hashtable getHash(byte type) 
{ 

if (type TYPE_SONG) 

return songs; 
else if (type = TYPE_ALBUM) 

return albums; 
else if (type = TYPE_ART1ST) 

return artists; 
else if (type = TYPE_AD) 

return ads; 
else if (type = TYPE_NEWS) 

return news; 
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else if (type = TYPE__TIP) 

return tips; 
return null; 

} 

public Object init(int ID, byte type) 
{ 

if (getHash(type).containsKey(new Integer(ID))) 
{ 

return get(ID, type); 

} 

else { 

return put(ID, type); 

} 

} 

public Object get(Integer ID, byte type) 
{ 

return (getHash(type)).get(ID); 

} 

public Object get(int ID, byte type) 
{ 

return get(new IntegerQD), type); • 

_ private Object makeNew(int ID, byte type) , 
{ 

if (type = TYPE_SONG) 

return new SongInfo(ID); 
else if (type = TYPE_ALBUM) 

return new Album Info(ID); 
else if (type = TYPE_ARTIST) 

return new ArtistInfo(ID); 
else if (type = TYPE_AD) 

return new Clip(ID, Clip.TYPE_AD); 
else if (type — TYPE_NEWS) 

return new Clip(ID, CJip.TYPE_NEWS); 
else if (type = TYPE_TIP) 

return new Clip(lD, ClipTYPEJTIP); 
return null; 

} 

private Object put(int ID, byte type) 
{ 

Hashtable hash = getHash(type); 
Object thing = makeNew(ID, type); 
hash.put(new Integer(ID), thing); 
return thing; 

} 

public Songlnfo randomSongQ 
{ 

long index = Util.random(songList length - 1); 
if (index > songList. length - 1 ) 

return null; 
return songList[(int) index]; 

} 

public Enumeration keys(byte type) 
{ 

if (type = TYPE_SONG) 

return songs.keys(); 
else if (type = TYPE_ALBUM) 

return albums.keysO; 
else if (type = TYPE ARTIST) 

return artists.keysO; 
else if (type = TYPE_AD) 

return ads.keysQ; 
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else if (type = TYPE_NEWS) 

return news.keysO; 
else if (type = TYPEJTIP) 

return tips.keysO; 
return null; 

} 

public int size(byte type) 
{ 

Hashtable hash = getHash(type); 
if (hash != null) 

return hash.sizeO; 

return 0; 

} 

private Clip[] getClipList(byte type) 
{ 

if(type = TYPE_AD) 

return adList; 
else if (type = TYPE_NEWS) 

return newsList; 
else if (type = TYPE_TIP) 

return tipList; ' 

^return mill; ~ "~ 

public Clip randomClip(byte type) 
{ 

ClipQ clips = getCHpList(type); 
if (clips — null || clips.length <= 0) 
return null; 

return clips [(int) Util.random(cIips. length - I)]; 

} 

public Vector randomClipList(byte type, short mediaType, int max) 
{ 

Vector list - new VectorO; 
Clip bip; 

// stop if we have enough or we've iterated too many times 
for (int i = 0; i < (max * 1 0) && HstsizeO < max; i++) 
{ 

int iterations = max; 
boolean cool = false; 
boolean done = false; 
do 
{ 

bip = randomClip(type); 
iterations—; 

// maybe we didn't get one 
if (bip = null) 
{ 

done = true; 

} 

else 
{ 

// we got one that fits! 
cool = (bip.media.inType(mediaType) && !Iist.contains(bip)); 
// we've got to stop sometime 
done - (cool || iterations < 0); 

} 

} 

while ('done); 

// if it was cool, go ahead 

if (cool) 

list.addElement(bip); 

} 
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return list; 

} 

} 

SonglnfoCachejava Page 9 of 9 1 1/05/99 t:32 PM 
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SonglnfoCacheUpdater 

package com.launch.PlaylistGenerator; 
import javax.servlet.http. H ttpServlet ; 
import java.util.Date; 

public class SonglnfoCacheUpdater extends Thread 
{ 

PlaylistGeneratorServIet servlet; 

public SongInfoCacheUpdater(PlaylistGeneratorServlet servlet) 
{ 

this.servlet = servlet; 

public void runO 
{ 

Thread.currentThreadOsetNameC'SonglnfoCacheUpdater"); 
// update every day 

long timeToSleep = Util.MILLISECONDSJN_SECOND * 

UtiLSECONDSJN_MINUTE * 
«, ~- Util.MINUTES IN HOUR ^ 



UtiI.HOURSJN_DAY; 



while (true) 
{ 



e.toStringO); 



try { Thread.sleep(timeToSleep); } catch (InterruptedException e) {}; 

try 
{ 

UtiLdebug("updating song cache at " + new DateO); 
Util.debugClast update was at " + serv!et.songCache.lastUpdate); 

// make a new cache 

SonglnfoCache cache = new SongInfoCache(null); 

// make sure to copy over the ratingsCache too! ! ! 
cache. ratingsCache = servletsongCache.ratingsCache; 

// install the new cache 
servIet.songCache - cache; 

Util.debug( M finished updating song cache at " + new DateO); 
Util.debug("last update is now at " + servIet.songCache. lastUpdate); 

} 

catch (Throwable e) 
{ 

System.err.println("SongInfoCacheUpdater caught an exception: " + 
e.printStackTraceO; 

> 



} 

} 

SonglnfoCacheUpdater.java Page 2 of 2 11/05/99 1:38 PM 
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SongList 



package com. launch.Play I istGenerator; 
import java.util. Vector, 
import java.util.Hashtable; 
5 import java.util.Enumeration; 

public class SongList implements Cloneable 
{ 

private Vector list = new Vector(); 
private Hashtable unique = new Hashtable(); 
to private boolean ordered = false; 

public SongListO 
{ 
} 

'** 

is * Creates a SongList from a Hashtable of songs 

V 

public SongList(Hashtable songs) 
{ 

Songlnfo info = null; 
20 Integer songID; 

^ , „ . f or (E numera t| 0rl e = songs.keysO; e.hasMoreElementsO;) " 

; { . . ■ v ; . .. , 

songID = (Integer) e.nextElementO; 
info = (Songlnfo) songs.get(songID); 
2s addElement(info); 
} 

} 

public SongList(Hashtable songs, short mediaType) 
{ 

30 Integer songID; 

Songlnfo info = null; 

for (Enumeration e = songs.keysO; e.hasMoreElementsO;) 
{ 

songID = (Integer) e.nextElementO; 
35 info = (Songlnfo) songs.get(songlD); 

if (info.media.inType(mediaType)) 

{ " 
addElement(info); 

} 

«0 } 

} 

public void addElement(SongInfo info) 
{ 

Integer ID - new lnteger(info.songID); 
4s // check unique constraint 

if (unique.get(ID) = null) 

{ 

Iist.addElement(info); 
unique.put(ID, info); 

» } 
} 

public void addElements(SongList list) 
{ 

if (list = null) 
55 return; 

for (int i = 0; i < list.sizeO; i++) 
{ 

addElement(list.e!ementAt(i)); 

} 

} 
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public void sort() 
{ 

sort(this, 0, list.sizeO - 1); 
ordered - true; 

} 

public int size() 
{ 

return list.sizeO; 

} 

public Songlnfo elementAt(int index) 
{ 

return (Songlnfo) listelementAt(index); 

} 

public void setSize(int newSize) 
{ 

Iist.setSize(newSize); 

} 

private void sort(SongList a, int from, int to) 
{ 

//quicksort 

~— — // if there is nothing to sortTretum"™^ ™. ^" ™ r «- 

. if ((a = null)||(a.size0<2))return; r I: . „ ^ _ „ . 

int i = from, j = to; 

Songlnfo center = a.elementAt((from + to) / 2); 
do{ 

while((i < to) && (center.comraRating < a.elementAt(i).commRating)) i++; 
while((j > from) && (center.commRating > a.elementAt(j).commRating)) j--; 

if(i<j){ • 

Songlnfo temp = a.elementAt(i); 

a.setEIementAt(a.elementAt(j), 0; 
a.setElementAt(temp, j); // swap elements 

, } 

if(i<=j){i++;j~;} 
} while(i<=j); 

if (from < j) sort(a, from, j); // recursively sort the rest 
if (i < to) sort(a, i, to); 

} 

public void setElementAt(Song!nfo info, int index) 
{ 

list.setElementAt(info, index); 

} 

public Songlnfo pickRandomO 
( 

if(size()<=0) 

return null; 
int lucky = (int) UtiI.random(sizeO - 1); 
if(lucky<0) 

return null; 
Songlnfo info - e!ementAt(lucky); 
Iist.removeElementAt(lucky); 
return info; 

} 

public Object cloneO 
{ 

SongList result = new SongListO; 
result.ordered = this.ordered; 
resultunique = (Hashtable) unique.cloneO; 
result.list = (Vector) listxloneO; 
return result; 

} 
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} 

SongListjava Page 3 of 3 1 1/05/99 1 :34 PM 
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SongRating 

package com.launch.PlaylistGenerator; 

public class SongRating 

{ 

public final static byte RATING_SOURCE_NONE = 0; 
public final static byte RATING_SOURCE_EXPLICIT = I ; 
public final static byte RATING_SOURCE_FROM_ A LB U M = 2; 
public final static byte RATING_SOURCE_FROM_ARTIST = 3; 

public final static byte RATING_SOURCE_AVERAGE__SONG_RATING_BY_ARTIST = 4; 

private short rating = (short) Constants.DEFAULTRATING; 
private boolean set = false; 
private byte type; 

public boolean isSetO 
{ 

return set; 

} 

— ~ public short set(short newRating, byte newType) — ~ ' — 

{ ; _ , : _ . _ . - 

rating — newRating; 
type = newType; 
set = true; 

return rating; 

public short get() 
{ 

return rating; 

> 

public byte getSourceO 
{ 

return type; 

} 

} 

SongRating.java Page 1 of 1 11/05/99 1:38 PM 
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Station 

package com.Iaunch.PlayiistGenerator; 

public class Station 

{ 

int ID; 

public Station(int station ID) 
{ 

ID = stationID; 

} 

} 

Station.java Page 1 of I 1 1/05/99 1:26 PM 
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StationList 

package com. launch. PlaylistGenerator, 
import java.util.Vector; 
public class StationList 

5 { 

private Vector slist; 

public StationListO 
{ 

slist = new VectorO; 

public Station stationAt(int i) 
« { 

return (Station) slist.elementAt(i); 

} 

public void addElement(Station s) 

—slistaddElementCs); " 1 . - ■~ r ™* .^ *~ rr ~s~~» ~ ~~ .. *< - ~~™' ■ : ^ 



— — . . 

public int sizeO ... ■ . 

{ 

return slist.sizeO; 

} 

public String inListO 

Integer HstQ = new IntegerfsizeQ]; 
int last = 0; 

for (int i ~ 0; i < slistsizeO; 
{ 

list[i] = new Integer(stationAt(i).ID); 

} 

return Util.join( N , list); 

} 

public Station get(int stationID) 

{ 

for (int i - 0; i < slistsizeO; i++) 
{ 

if (stationAt(i).ID = stationID) 
{ 

return stationAt(i); 

> 

return null; 

} 

} 

StationListjava Page 1 of 1 1 1/05/99 1 :26 PM 
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Util 

package com.launch.PlaylistGenerator; 

import java.io.OutputStream; 

import java.util.Date; 

import javax.servlet.ServletOutputStream; 

import java.io.IOException; 

public class Util 

{ 

public static final int MILLISECONDS_IN_SECOND = 1000; 

public static final int SECONDS JNJvirNUTE = 60; 

public static final int MINUTES_IN_HOUR = 60; 

public static final int HOURS_IN_DAY = 24; 

public static final int DAYS_IN_WEEK = 7; 

public static final int DAYS JNMONTH - 30; 

public static final int DISPLAY_TEXT = 0; 

public static final int DISPLAY HTML = 1 ; 

public static final String newLine = u \r\n"; 

public static final short average(double count, double sum) 

{ 

if (count = 0) 

return (short) Math.round(sum / count); 

> 

public static final long random(int ceiling) 
{ 

return Math.round(Math.randomO * ceiling); 

} 

public static final String join (String delim, Object values[]) 
{ 

String result - ""; 
int i = 0; 

for (; i < values.length; i++) . 

result = result.concat(values[i].toStringO + delim); 

if(i>0) 

result = resultsubstring(0, (result. lengthO - delim.lengthO)); 
return result; 

} 

public static final String fix(double number, int precision, int zeroFill) 
{ 

double power = Math.pow( 10, precision); 

double fixed = Mathxound(number * power) / power; 

String mantissa = new Long(Math.round(fixed)).toStringO; 

String result = mantissa; 

for (int i = mantissa.length(); i < zeroFill; i++) 

result = new String( H 0" + result); 
return result; 

} 

public static final void out(ServletOutputStream stream, String whatever) 
{ 

try 
{ 

if (stream = null) 

System.out.printIn(whatever); 

else 

stream.println(whatever); 

} 

catch (lOException e) 

{ 

} 

} 
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public static final void debug(String info) 

{ 

System.out.println(info); 

} 

public final static String tab(int times) 
{ 

String result = 

for (int i = 0; i < times; H-+) 

{ 

result = resu!t.concat(" "); 

} 

return result; 

} 

public static final void markQueryFinished(String threadName, Date startDate) 
{ 

Util.debug(newLine + threadName + w started getting data after H 

+ ((new DateO-getTimeO - StartDate.getTimeO) / 1 000.0) 
+ H seconds" + newLine); 

^public^statiFMS^id^ 

Util.debug(newLine + new DateO.toStringO + ■" " + threadName + n took " 

+ ((new DateO-getTimeO - StartDate.getTimeO) / ! 000.0) 
+ H seconds" + newLine); 

} 

public static final String tabO 
{ 

return tab(l); 

} 

} 

Util.javaPage3of3 1 1/05/99 1:37 PM 
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package com.launch.PIaylistGenerator; 

public class WeightMatrix 

{ 

public final static byte RATING = 0; 

public final static byte DJS = 1 ; 

public final static byte NETP = 2; 

public final static byte COM M_RATING = 3; 

public final static byte LAST_PLAYED = 4; 

public final static byte BDS = 5; 

public final static byte CONFIDENCE =6; 

// rating, djs, netp, commRating, lastPIayed, bds, conf 

public double matrix[][] = { 

{0.00, 0.33, 0.00, 0.10, 0.25, 0.20, 0.0}, // no rating 
{0.70, 0.00, 0.00, 0.00, 0.30, 0.00, 100.0}, // explicit rating 
{0.45, 0.05, 0.00, 0.05, 0.20> 0.20, 50.0}, // album rating only 
{0.40, 0.10, 0.00, 0,05, 0.20, 0.20, 30.0}, // artist only 
{0.35, 0. 1 5, 0.00, 0.05, 0.20, 0.20, 20.0} // cross-propagated 



song ratings 



}; 



} 

-WeightMatrixjava Page 1 of 1— — 1 1/05/99 1:32 PM~ 



App. 2-194 



WO 01/35667 



219 



PCT/US00/30919 



CLAIMS 



What is claimed is: 

1 . A method for broadcasting data streams through a computer network to a user's computer, the steps 
2 comprising: 

providing a database of data streams; 
4 selecting a data stream according to a selection method; 

transmitting one of said data streams to the user's computer; 
6 receiving feedback expressing a preference from the user regarding said transmitted data stream; 

and 

s updating said selection method to better reflect said preference of the user; whereby 

data streams transmitted to the user are biased according to said preference. 

2. The method for broadcasting data streams through a computer network to a user's computer of Claim 
i "further comprising: ~ .«^-^-~™~ „ ^ — ,„ — — — 

said selection method including generating a list of data streams to transmit to the user's computer; 
4 transmitting one of said listed data streams to the user's computer, and 

updating said list of data streams to better reflect said preference of the user; whereby 
6 data streams transmitted to the user are biased according to said preference. 

3. The method for broadcasting data streams through a computer network of Claim 1 , the steps further 
2 comprising: 

receiving feedback expressing preferences from sources other than the user. 

4 . The method for broadcasting data streams through a computer network of Claim 3, wherein the step of 
2 receiving preferences from sources other than the user further comprises: 

receiving feedback expressing preferences from the group consisting of other users, commercial 
4 radio stations, and lists of popular songs. 

5. The method for broadcasting data streams through a computer network of Claim 1 , further comprising: 
2 informing the user generally regarding said database and said data streams; 

querying the user as to data stream preference prior to generating an initial transmission list of data 
4 streams; whereby 

said initial list reflects general preferences of the user. 



6. The method for broadcasting data streams through a computer network of Claim 1 , wherein said data 
2 streams are selected from the group consisting of songs and videos. 

7. The method for broadcasting data streams through a computer network of Claim I, wherein said 
2 transmitted data stream is removed from said transmission list. 

8. The method for broadcasting data streams through a computer network of Claim 7, wherein said data 
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2 stream removed from said transmission list is listed on a transmitted data stream list. 

9. The method for broadcasting data streams through a computer network of Claim I , wherein said step 
2 of transmitting one of said data streams further comprises transmitting said one of said data streams in 

conformance with applicable copyright law. 

10. The method for broadcasting data streams through a computer network of Claim 9, wherein said 
: conformance with applicable copyright law applies to all transmitted datastreams. 

11. A data stream system for providing preferred data streams to a user, comprising: 

2 a connection to a computer network, said computer network connected to a computer of the user; 

a database of data streams, said database available to said computer network; 
4 a data stream controller, said data stream controller transmitting data streams to said user's 

computer according to a selection program; 
6 a user interface, said user interface coupled to said user's computer and receiving said data streams 

for the user and providing a feedback mechanism for the user so that the user may indicate a preference 
6 regarding data streams transmitted by said data stream controller; 

said selection program receiving indications from the user, said selection program modifying its 
to selection of data streams for transmission to said user's computer according to said user preference; 

whereby 

12 data streams selected by said selection program are biased according to said user preference. 

12. The data stream system for providing preferred data streams to a user of Claim 1 1, wherein said 
2 computer network comprises the Internet. 

.13. The data stream system for providing preferred data streams to a user of Claim 1 1, wherein said 

2 database is a song database and the data streams are songs. 

14. The data stream system for providing preferred data streams to a user of Claim 1 1, wherein said 
2 database is a music video database and the data streams are music videos. 

1 5. The data stream system for providing preferred data streams to a user of Claim 1 1 , wherein said user 
2 interface comprises an electronic media player. 

16. The data stream system for providing preferred data streams to a user of Claim 15, wherein said 
2 electronic media player is selected from the group consisting of RealPlayer, Apple QuickTime, and Windows 

Media Player. 



17. The data stream system for providing preferred data streams to a user of Claim II, wherein said 

2 selection program creates a list of data streams for transmission to the user. 
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1 8. The data stream system for providing preferred data streams to a user of Claim 1 7, wherein said 
2 selection program modifies said list of data streams for transmission to the user according to said user preference. 

1 9. The data stream system for providing preferred data streams to a user as set forth in Claim 1 1 , further 
2 comprising: 

said data stream controller transmitting said data streams in compliance with applicable copyright 

4 law. 

20. The data stream system for providing preferred data streams to a user as set forth in Claim 1 9, further 

2 comprising: 

said data stream controller transmitting all data streams in compliance with applicable copyright 

4 law. 



— - 2i. — - A user interface for an Internet datastream transmission system, comprising: : - ^ - ^ - — 

2 a media player, said playing data streams; 

a rating tool, said rating tool indicating a rating for a data stream currently played by said media 
4 player; and 

a data stream information display, said data stream information display displaying information for 
6 said data stream currently played by said media player, whereby 

a user can indicate a preference regarding said data stream currently played by said media player. 

22. A user interface for an Internet datastream transmission system as set forth in Claim 21, further 
2 comprising: 

a playlist generator, said playlist generator generating playlists of data streams for said media 
4 player, said playlist generator selecting data streams according to preferences indicated by said user. 

23. A user interface for an Internet datastream transmission system as set forth in Claim 22, further 
2 comprising: 

said data streams selected by said playlist generator being in compliance with applicable 
4 copyright law. 
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